【Spring笔记】05_声明式事务控制

本博文笔记整理自黑马课程《SSM框架Spring+SpringMVC+MyBatis全覆盖》

1 编程式事务控制相关对象

1.1 PlatformTransactionManager

PlatformTransactionManager 接口是spring 的事务管理器,它里面提供了我们常用的操作事务的方法。

方法说明
TransactionStatus getTransation(TransactionDefinition def)获取事务状态信息
void commit(TransactionStatus status)提交事务
void rollback(TransactionStatus status)回滚事务

🚯注意

PlatformTransactionManager 是接口类型,不同的Dao 层技术则有不同的实现类,例如:Dao 层技术是jdbc 或mybatis 时:org.springframework.jdbc.datasource.DataSourceTransactionManager
Dao 层技术是hibernate时:org.springframework.orm.hibernate5.HibernateTransactionManager

1.2 TransactionDefination

TransactionDefinition 是事务的定义信息对象,里面有如下方法:

方法说明
int getIsolationLevel()获得事务的隔离级别
int getPropogationBehavior()获得事务的传播行为
int getTimeout()获得超时时间
boolean isReadOnly()是否只读
1.2.1 事务隔离级别

事务隔离级别:一个事务对数据库的修改与并行的另一个事务的隔离程度。

设置隔离级别,可以解决事务并发产生的问题,如脏读、不可重复读和虚读(幻读)。

两个并发事务同时访问数据库表相同的行时,可能存在以下三个问题:
1、幻读:事务T1读取一条指定where条件的语句,返回结果集。此时事务T2插入一行新记录,恰好满足T1的where条件。然后T1使用相同的条件再次查询,结果集中可以看到T2插入的记录,这条新纪录就是幻想。
2、不可重复读取:事务T1读取一行记录,紧接着事务T2修改了T1刚刚读取的记录,然后T1再次查询,发现与第一次读取的记录不同,这称为不可重复读。
3、脏读:事务T1更新了一行记录,还未提交所做的修改,这个T2读取了更新后的数据,然后T1执行回滚操作,取消刚才的修改,所以T2所读取的行就无效,也就是脏数据。

  • ISOLATION_DEFAULT(数据库默认隔离级别)
  • ISOLATION_READ_UNCOMMITTED(读未提交): 幻想读、不可重复读和脏读都允许。
  • ISOLATION_READ_COMMITTED(读已提交): 允许幻想读、不可重复读,不允许脏读
  • ISOLATION_REPEATABLE_READ(可重复读): 允许幻想读,不允许不可重复读和脏读
  • ISOLATION_SERIALIZABLE(串行化): 幻想读、不可重复读和脏读都不允许

首先要明确Oracle里支持的隔离级别:read committed/serializable,默认的是read committed,而MySQl支持的隔离级别:read uncommitted/read committed/repeatable read/serializable,默认的是repeatable read;

1.2.2 事务传播行为
  • REQUIRED:如果当前没有事务,就新建一个事务,如果已经存在一个事务中,加入到这个事务中。一般的选择(默认值)

  • SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方式执行(没有事务)

  • MANDATORY:使用当前的事务,如果当前没有事务,就抛出异常

  • REQUERS_NEW:新建事务,如果当前在事务中,把当前事务挂起。

  • NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,就把当前事务挂起

  • NEVER:以非事务方式运行,如果当前存在事务,抛出异常

  • NESTED:如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行REQUIRED 类似的操作

  • 超时时间:默认值是-1,没有超时限制。如果有,以秒为单位进行设置

  • 是否只读:建议查询时设置为只读

1.3 TransactionStatus

TransactionStatus 接口提供的是事务具体的运行状态,方法介绍如下。

方法说明
boolean hasSavepoint()是否存储回滚点
boolean isCompleted()事务是否完成
boolean isNewTransaction()是否是新事务
boolean isRollbackOnly()事务是否回滚

2 基于XML 的声明式事务控制

2.1 什么是声明式事务控制

Spring 的声明式事务顾名思义就是采用声明的方式来处理事务。这里所说的声明,就是指在配置文件中声明,用在Spring 配置文件中声明式的处理事务来代替代码式的处理事务。

声明式事务处理的作用
  • 事务管理不侵入开发的组件。具体来说,业务逻辑对象就不会意识到正在事务管理之中,事实上也应该如此,因为事务管理是属于系统层面的服务,而不是业务逻辑的一部分,如果想要改变事务管理策划的话,也只需要在定义文件中重新配置即可
  • 在不需要事务管理的时候,只要在设定文件上修改一下,即可移去事务管理服务,无需改变代码重新编译,这样维护起来极其方便

注意:Spring 声明式事务控制底层就是AOP。

2.2 声明式事务控制的实现

声明式事务控制明确事项:
 谁是切点?
 谁是通知?
 配置切面?

  1. 引入tx命名空间

    在这里插入图片描述

  2. 配置事务增强

        <!--配置平台事务管理器-->
        <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
            <property name="dataSource" ref="dataSource"/>
        </bean>
    
        <!--通知  事务的增强-->
        <tx:advice id="txAdvice" transaction-manager="transactionManager">
            <!--设置事务的属性信息的-->
            <tx:attributes>
                <tx:method name="transfer" isolation="REPEATABLE_READ" propagation="REQUIRED" read-only="false"/>
                <tx:method name="save" isolation="REPEATABLE_READ" propagation="REQUIRED" read-only="false"/>
                <tx:method name="findAll" isolation="REPEATABLE_READ" propagation="REQUIRED" read-only="true"/>
                <tx:method name="update*" isolation="REPEATABLE_READ" propagation="REQUIRED" read-only="true"/>
                <tx:method name="*"/>
            </tx:attributes>
        </tx:advice>
    
  3. 配置事务AOP 织入

        <!--配置事务的aop织入-->
        <aop:config>
            <aop:pointcut id="txPointcut" expression="execution(* com.itheima.service.impl.*.*(..))"/>
            <aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
        </aop:config>
    
  4. 测试事务控制转账业务代码

        public void transfer(String outMan, String inMan, double money) {
            accountDao.out(outMan,money);
            int i = 1/0;
            accountDao.in(inMan,money);
        }
    

2.3 切点方法的事务参数的配置

	<!--事务增强配置-->
    <tx:adviceid="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:methodname="*"/>
        </tx:attributes>
    </tx:advice>

其中,<tx:method> 代表切点方法的事务参数的配置,例如:
<tx:method name=“transfer” isolation=“REPEATABLE_READ” propagation=“REQUIRED” timeout="-1" read-only=“false”/>
 name:切点方法名称
 isolation:事务的隔离级别
 propogation:事务的传播行为
 timeout:超时时间
 read-only:是否只读


3 基于注解的声明式事务控制

3.1 使用注解配置声明式事务控制

  1. 编写AcountDao

    @Repository("accountDao")
    public class AccountDaoImpl implements AccountDao {
    
        @Autowired
        private JdbcTemplate jdbcTemplate;
    
        public void out(String outMan, double money) {
            jdbcTemplate.update("update account set money=money-? where name=?",money,outMan);
        }
    
        public void in(String inMan, double money) {
            jdbcTemplate.update("update account set money=money+? where name=?",money,inMan);
        }
    }
    
  2. 编写AccoutService

    @Service("accountService")
    @Transactional(isolation = Isolation.REPEATABLE_READ)
    public class AccountServiceImpl implements AccountService {
    
        @Autowired
        private AccountDao accountDao;
    
        @Transactional(isolation = Isolation.READ_COMMITTED,propagation = Propagation.REQUIRED)
        public void transfer(String outMan, String inMan, double money) {
            accountDao.out(outMan,money);
            int i = 1/0;
            accountDao.in(inMan,money);
        }
    
        //@Transactional(isolation = Isolation.DEFAULT)
        public void xxx(){}
    }
    
  3. 编写applicationContext.xml 配置文件

        <!--组件扫描-->
        <context:component-scan base-package="com.itheima"/>
        
        <!--事物的注解驱动-->
        <tx:annotation-driven transaction-manager="transactionManager"/>
    

3.2 注解配置声明式事务控制解析

① 使用@Transactional 在需要进行事务控制的类或是方法上修饰,注解可用的属性同xml 配置方式,例如隔离级别、传播行为等。
② 注解使用在类上,那么该类下的所有方法都使用同一套注解参数配置。
③ 使用在方法上,不同的方法可以采用不同的事务参数配置。
④ Xml配置文件中要开启事务的注解驱动<tx:annotation-driven />

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值