@Transactional是我们项目中很常用的一个注解了,@Transactional可以为我们控制事务,可以保证事务内的操作要么全部成功,要么全部失败,@Transactional也是基于AOP切面实现事务的功能的。
下面列举一下使用@Transactional会导致事务失效的场景:
1、@Transactional应用在了非public的方法上,事务将会失效:
原因是SpringAOP对事务进行代理增强的时候会调用到computeTransactionAttribute方法,这个方法里面进行了判断,会检查目标方法是否是publice修饰,如果不是public修饰,则事务不会生效;
protected TransactionAttribute computeTransactionAttribute(Method method,
Class<?> targetClass) {
// Don't allow no-public methods as required.
if (allowPublicMethodsOnly() && !Modifier.isPublic(method.getModifiers())) {
returnnull;
}
2、在同一个类中调用@Transactional注解的方法会失效:
比如有一个UserManager类,里面有一个方法A,一个方法B,方法B也加了@Transactional注解,如果在方法A中直接调用方法B,B的事务是不会生效的,原因是AOP,AOP只有方法在被类对象调用的时候才会增强;
3、没有设置 rollbackFor或者 rollbackFor设置错误导致事务失效:
rollbackFor是用来指定遇到哪种异常时,对事务进行回滚,如果不设置,则遇到异常不会回滚,一般情况都要设置成@Transactional(rollbackFor = Exception.class),遇到任何异常都回滚;
4、设置了 rollbackFor却被catch了,导致事务失效:
@Transactional(rollbackFor = Exception.class)正确设置了rollbackFor,可是却在代码中进行了try catch,导致异常不能抛出,从而导致事务失效;
例如:
@Transactional(rollbackFor = Exception.class)
public void save(User user){
try(
service1.save(user);
service2.update(user.getId());
)catch(Exception e){
// 这里对Excaption进行了catch,导致异常无法抛出,事物就不会生效,正确做法是直接抛出异常
e.printStackTrace();
}
}
5、@Transactional的propagation属性设置错误,导致事物失效:
propagation如果配置为如下三种,很容易导致事务失效:
ransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。