Spring 事务
Spring 事务的管理方式
- 编程式事务:在代码中硬编码(不推荐)
- 声明式事务:在配置文件中配置(推荐)
声明式事务又分为2种:
- 基础xml的声明式事务
- 基于注解的声明式事务
Spring 事务的隔离级别
TransactionDefinition 接口中定义了五个表示隔离级别的常量:
- TransactionDefinition.ISOLATION_DEFAULT使用后端数据库默认的隔离级别,MySQL默认采用repeatable_read隔离级别 Oracle 默认采用 Read_Committed 隔离级别
- TransactionDefinition.ISOLATION_READ_UNCOMMITTED:最低隔离级别,允许读取未提交的数据,可能会导致脏读、不可重复读、幻读
- TransactionDefinition.ISOLATION_READ_COMMITTED:语序读取已提交的数据,可以阻止脏读,但是不可重复读和幻读仍有可能发生
- 可重复读
- TransactionDefinition.ISOLATION_SERIALIZABLE:序列化,最高隔离级别,完全服从 ACID 的隔离级别。所有事务依次执行互不干扰,可以防止脏读、不可重复读、幻读,这将严重影响性能,通常情况下不会用到这个级别。
Spring事务中有哪几种传播行式
- TransactionDefinition.PROPAGATION_REQUIRED(必须):
- 如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
- 我们平时使用@Transaction 默认就是使用这个事务。
- TransactionDefinition.PROPAGATION_REQIURED_NEW:
- 创建新的事务,如果当前存在事务则将事务挂起,也就是不管外部是否有事务,都将开启新的事务,各自互不相干;
- TransactionDefinition.PROPAGATION_NESTED(嵌套):
- 如果当前存在事务,则创建一个事务作为当前事务的嵌套事务,如果当前不存在事务,则等价于PROPAGATION_REQUIRED事务
- TransactionDefinition.PROPAGATION_MANDATORY:
- 如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。(mandatory:强制性)
若是错误的配置以下 3 种事务传播行为,事务将不会发生回滚:
- TransactionDefinition.PROPAGATION_SUPPORTS: 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
- TransactionDefinition.PROPAGATION_NOT_SUPPORTED: 以非事务方式运行,如果当前存在事务,则把当前事务挂起。
- TransactionDefinition.PROPAGATION_NEVER: 以非事务方式运行,如果当前存在事务,则抛出异常。
方法A调用方法B,
- A方法加了事务,则a,b在同事务中;
- B方法加了事务,a,b为同一类,事务失效;a,b不同类,则B有事务。
原理:
spring 在扫描bean的时候会扫描方法上是否包含@Transactional注解,如果包含,spring会为这个bean动态地生成一个子类(即代理类,proxy),代理类是继承原来那个bean的。
此时,当这个有注解的方法被调用的时候,实际上是由代理类来调用的,代理类在调用之前就会启动transaction。然而,如果这个有注解的方法是被同一个类中的其他方法调用的,那么该方法的调用并没有通过代理类,而是直接通过原来的那个bean,所以就不会启动transaction,我们看到的现象就是@Transactional注解无效。
那回到一开始的问题,我们调用的方法A不带注解,因此代理类不开事务,而是直接调用目标对象的方法。当进入目标对象的方法后,执行的上下文已经变成目标对象本身了,因为目标对象的代码是我们自己写的,和事务没有半毛钱关系,此时你再调用带注解的方法,照样没有事务,只是一个普通的方法调用而已。
简单来说,内部调用本类方法,不会再走代理了,所以B的事务不起作用。
如果AB不同类,A方法调用了其他类的方法B(方法有加事务),故B有事务。
@Transactional(rollbackFor = Exception.class)注解
当@Transaction 注解作用于类上时,该类的所有 public 方法都将具有该类的属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义。如果类或者方法加了这个注解,那么这个类里面的方法会抛出异常,就会回滚,数据库里的数据也会回滚。
在@Transaction注解中如果不配置 rollbackFor 属性,那么事务只会遇到 RuntimeException 的时候才会回滚,加上rollbackFor = Exception.class,可以让事务在遇上非运行时异常时也会回滚。