Day23 Spring事务 Xml配置和注解配置***

Spring事务管理-准备工作

  • (1)spring的事务实现方式:
    1.编程式事务管理
    2.声明式事务管理 (使用aop xml 或注解的配置即可实现事务管理)
  • (2)准备数据库
  • (3)POM.xml

sql

create database day23db ;
use day23db;
create table `account` (
  `id` int(8) not null  primary key auto_increment,
  `name` varchar(64) default null,
  `money` double default null
) ;

insert into account values(null,'jack',1000);
insert into account values(null,'rose',1000);

# 转账
update account set money=money-500 where `name`='jack';
update account set money=money+500 where `name`='rose';

pom.xml

   <dependencies>
        <!-- jdbc-->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>5.1.46</version>
        </dependency>
        <!-- ioc-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.2.9.RELEASE</version>
        </dependency>
        <!-- jdbctemplate-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-jdbc</artifactId>
            <version>5.2.9.RELEASE</version>
        </dependency>
        <!-- test-->
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-test</artifactId>
            <version>5.2.9.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
        </dependency>
       <!-- aop-->
        <dependency>
            <groupId>org.aspectj</groupId>
            <artifactId>aspectjweaver</artifactId>
            <version>1.7.2</version>
        </dependency>
    </dependencies>

测试类TestAccountService

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class TestAccountService {
    
    
    @Autowired
    IAccountService service ;
    @Test
    public void test01(){
    
    
        //业务对象
        //IAccountService service = new AccountServiceImpl();
        //调用方法 translate(一个人转出,另一个转入,金额)
        service.translate("jack","rose",500.00);
    }
}

IAccountService

public interface IAccountService {
    
    
    //转账本身就是从一个账户减钱,另一个账户加钱
    void translate(String from, String to, double money);
}

AccountServiceImpl

@Service
public class AccountServiceImpl implements IAccountService{
    
    
    @Autowired
    IAccountDao dao ;

    @Override
    public void translate(String from, String to, double money) {
    
    
        //创建AccountDaoImpl
        //AccountDaoImpl dao = new AccountDaoImpl();
        //调用方法
        dao.translateOut(from,money);
        //除0异常
        System.out.println(1/0); 
        dao.translateIn(to,money);
        System.out.println("--translate");
    }
}

IAccountDao

public interface IAccountDao {
    
    
	//标记为过时
    @Deprecated
    void update(String account, double v);
    void translateOut(String account, double v);
    void translateIn(String account, double v);
}

AccountDaoImpl

@Repository
public class AccountDaoImpl implements IAccountDao {
    
    
    @Autowired
    JdbcTemplate jdbcTemplate;
    @Override
    public void update(String account, double v) {
    
    
       System.out.println(account+" "+v);
       jdbcTemplate.update("update account set money=money+? where `name`=?",v,account);
    }

    @Override
    public void translateOut(String account, double v) {
    
    
        jdbcTemplate.update("update account set money=money-? where `name`=?",v,account);
    }

    @Override
    public void translateIn(String account, double v) {
    
    
        jdbcTemplate.update("update account set money=money+? where `name`=?",v,account);
    }
}

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context.xsd http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop.xsd">

    <!-- 数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/day23db"/>
        <property name="username" value="root"/>
        <property name="password" value=""/>
    </bean>
    <!-- jdbcTemplate-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <constructor-arg name="dataSource" ref="dataSource"/>
    </bean>

    <!--包扫描创建对象 扫描包与它下面的子包中的所有类-->
    <context:component-scan base-package="cn.cyl"/>
    
</beans>

Spring事务管理-转账-没有事务

AccountServiceImpl中 transfer 方法中 int i = 1/0 出现了除0异常;发现问题之前,转出成功。发现问题之后,转入失败,没有收到事务控制,导致数据前后不一致
在这里插入图片描述

Spring事务管理-转账-xml事务

解决方式是使用spring基于xml的事务管理.
步骤:
一.在bean.xml中配置3个内容
1.配置平台事务管理器(注入dataSource)
2.编写一个tx:Advice 设置事务的增强
3.配置aop (配置切点, 切点和增强的组合);
在这里插入图片描述

因为要使用 tx 标签,要添加 tx 约束。
applicationContext.xml

扫描二维码关注公众号,回复: 12010864 查看本文章
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context" xmlns:tx="http://www.springframework.org/schema/tx"
       xmlns:aop="http://www.springframework.org/schema/aop"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context.xsd
         http://www.springframework.org/schema/tx
         http://www.springframework.org/schema/tx/spring-tx.xsd
          http://www.springframework.org/schema/aop
           http://www.springframework.org/schema/aop/spring-aop.xsd">


<!-- 数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/day23db"/>
        <property name="username" value="root"/>
        <property name="password" value=""/>
    </bean>
    <!-- jdbcTemplate-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <constructor-arg name="dataSource" ref="dataSource"/>
    </bean>

    <!--包扫描创建对象 扫描包与它下面的子包中的所有类-->
    <context:component-scan base-package="cn.cyl"/>

    <!--1.要在bean.xml中创建spring的平台事务管理器  (DataSourceTransactionManager   )  注入dataSource-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>

    <!--2.我们需要创建一个 tx:advice 增强-->
    <!--里面定义了筛选那些方法进行事务控制, 设置事务的隔离级别规则,  设置事务传播行为的规则  , 设置事务超时时间  设置事务是否只读..-->
    <tx:advice id="adviceId" transaction-manager="transactionManager">
        <tx:attributes>
            <!--
                该增强中的规则:
                name代表筛选那些方法收到事务的增强
                isolation : 隔离级别 REPEATABLE_READ 可重复读
                propagation: 传播行为
                read-only: 是否为只读事务
                timeout: 事务是否超时
            -->
            <tx:method name="translate" isolation="REPEATABLE_READ"  propagation="REQUIRED" read-only="false" timeout="-1" />
        </tx:attributes>
    </tx:advice>

    <!--3.配置aop-->
    <aop:config>
        <!--3.1 配置切点-->
        <aop:pointcut id="translate" expression="execution(* cn.cyl.service.AccountServiceImpl.translate(..))"/>
        <!--3.2 配置切点和增强的组合-->
        <aop:advisor advice-ref="adviceId" pointcut-ref="translate"/>
    </aop:config>

</beans>

Spring事务管理-转账-注解

使用注解事务管理:

  1. 重新复制 dao层的代码 service层的代码 (给实现类上添加@Repository @Service) 注入的时候使用@Autowried自动注入
  2. 重新编写一个applicationContext.xml (保留: 加载db.properties , DataSource , JdbcTemplate , 平台事务管理器其他xml配置删除);先配置包扫描,再给该xml中重新添加 注解驱动 (支持注解方式的事务)
  3. 给AccountServiceImpl 类上添加一个@Transactional注解即可。
    给某个方法上添加表示这个方法有事务,类上添加表示该类下所有方法都有事务。
  4. 测试AccountService 的transfer方法 看看是否能够控制事务。

applicationContext.xml

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xmlns:context="http://www.springframework.org/schema/context"
       xmlns:tx="http://www.springframework.org/schema/tx"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd
        http://www.springframework.org/schema/context
         http://www.springframework.org/schema/context/spring-context.xsd
         http://www.springframework.org/schema/tx
         http://www.springframework.org/schema/tx/spring-tx.xsd">

<!-- 数据源-->
    <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/day23db"/>
        <property name="username" value="root"/>
        <property name="password" value=""/>
    </bean>
    <!-- jdbcTemplate-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <constructor-arg name="dataSource" ref="dataSource"/>
    </bean>

    <!--包扫描创建对象 扫描包与它下面的子包中的所有类-->
    <context:component-scan base-package="cn.cyl"/>

    <!--1.要在bean.xml中创建spring的平台事务管理器  (DataSourceTransactionManager   )  注入dataSource-->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"/>
    </bean>

    <!--2.配置注解驱动-->
    <tx:annotation-driven transaction-manager="transactionManager"/>

</beans>

AccountServiceImpl

@Service
//类上添加
//@Transactional(isolation = Isolation.REPEATABLE_READ ,propagation = Propagation.REQUIRED ,timeout = -1)
public class AccountServiceImpl implements IAccountService{
    
    
    @Autowired
    IAccountDao dao ;
	
	//方法上添加
	@Transactional(isolation = Isolation.REPEATABLE_READ ,propagation = Propagation.REQUIRED ,timeout = -1)
    @Override
    public void translate(String from, String to, double money) {
    
    
        //创建AccountDaoImpl
        //AccountDaoImpl dao = new AccountDaoImpl();
        //调用方法
        dao.translateOut(from,money);
        //除0异常
        System.out.println(1/0); 
        dao.translateIn(to,money);
        System.out.println("--translate");
    }
}

Spring事务管理-转账-有事务

AccountServiceImpl中 transfer 方法中 int i = 1/0 出现了除0异常;发现问题之前,转出成功。发现问题之后,因为有事务控制,于是进行回滚,让转出操作退回到原数据,这样就让数据前后一致
在这里插入图片描述

猜你喜欢

转载自blog.csdn.net/qq_43639081/article/details/109140184
今日推荐