@Transactional 事务注解原理
@Transactional
的工作机制是基于 AOP 实现的,AOP 又是使用动态代理实现的。如果目标对象实现了接口,默认情况下会采用 JDK 的动态代理,如果目标对象没有实现了接口,会使用 CGLIB 动态代理。
如果一个类或者一个类中的 public 方法上被标注@Transactional
注解的话,Spring 容器就会在启动的时候为其创建一个代理类。
在调用被@Transactional
注解的 public 方法的时候,实际调用的是TransactionInterceptor
类中的 invoke()
方法。这个方法的作用就是在目标方法之前开启事务,方法执行过程中如果遇到异常的时候回滚事务,方法调用完成之后提交事务。
作用范围
- 当用作类上时,该类上的所有 public 方法将都具有该类型的属性。
- 当用作方法上时,该方法所在类上的注解将失效,该注解只能应用在 public 方法上。
- 不建议用在接口或接口方法上,因为这只有在使用基于接口的动态代理是才会生效
!注意:只有来自外部的方法调用才会引起事务行为,类内部方法调用本类内部的其他方法并不会引起事务行为。
注解属性
- value属性:当配置多个事务管理器时,可以使用该属性指定选择哪个事务管理器。
- propagation属性:用于指定事务的传播行为、默认值为 REQUIRED。
属性 | 含义 |
---|---|
REQUIRED | 如果当前没有事务,就新建一个事务,如果已经存在一个事务,则加入到这个事务中 |
SUPPORTS | 支持当前事务,如果当前没有事务,就以非事务方式执行 |
MANDATORY | 表示当前方法必须在事务中运行。如果当前事务不存在,则会抛出一个异常 |
REQUIRES_NEW | 表示当前方法必须运行在它自己的事务中(一个新的事务将被启动)。如果存在当前事务,在该方法执行期间,当前事务会被挂起 |
NOT_SUPPORTED | 表示该方法不运行在事务中。如果当前存在事务,就把当前事务挂起 |
NEVER | 表示当前方法不应该运行在事务上下文中。如果当前正有一个事务在运行,则会抛出异常 |
NESTED | 如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则执行与REQUIRED类似的操作 |
- isolation属性:用于指定事务的隔离规则,默认值为DEFAULT
属性 | 含义 |
---|---|
DEFAULT | 使用后端数据库默认的隔离级别 |
READ_UNCOMMITTED | 允许读取尚未提交的数据变更(最低的隔离级别) |
READ_COMMITTED | 允许读取并发事务已经提交的数据 |
REPEATABLE_READ | 对同一字段的多次读取结果都是一致的,除非数据是被本身事务自己所修改 |
SERIALIZABLE | 完全服从ACID的隔离级别,也是最慢的事务隔离级别,因为它通常是通过完全锁定事务相关的数据库表来实现的(最高的隔离级别) |
- timeout属性:事务的超时时间,默认值为-1(不会超时)。如果超过该时间限制但事务还没有完成,则自动回滚事务。
- readOnly属性:指定事务是否为只读事务,默认值为 false。
- rollbackFor属性、rollbackForClassName属性:用于指定能够触发事务回滚的异常类型,并且可以指定多个异常类型。
- noRollbackFor属性、noRollbackForClassName属性:用于设置哪些异常不需要回滚
Spring AOP 自调用问题
若同一类中的其他没有 @Transactional
注解的方法内部调用有 @Transactional
注解的方法,有@Transactional
注解的方法的事务会失效。
这是由于Spring AOP代理的原因造成的,因为只有当 @Transactional
注解的方法在类以外被调用的时候,Spring 事务管理才生效。
MyService
类中的method1()
调用method2()
就会导致method2()
的事务失效。
@Service
public class MyService {
private void method1() {
method2();
//......
}
@Transactional
public void method2() {
//......
}
}
解决办法就是避免同一类中自调用或者使用 AspectJ 取代 Spring AOP 代理。
@Transactional 的使用注意事项总结
@Transactional
注解只有作用到 public 方法上事务才生效,不推荐在接口上使用;- 避免同一个类中调用
@Transactiona
l 注解的方法,这样会导致事务失效; - 正确的设置
@Transactional
的rollbackFor
和propagation
属性,否则事务可能会回滚失败; - 被
@Transactional
注解的方法所在的类必须被 Spring 管理,否则不生效; - 底层使用的数据库必须支持事务机制,否则不生效;