springboot 注解@Transactional失效的原因
springboot 注解@Transactional失效的原因参考源码分析https://blog.csdn.net/oyc619491800/article/details/105826597一:声明式事务@Transactional二:注解@Transactional使用三:常用,使用该注解时,记得加异常回滚,readOnly 属性readOnly :指定事务是否为只读事务,默认值为 fal
springboot 注解@Transactional失效的原因
参考源码分析
https://blog.csdn.net/oyc619491800/article/details/105826597
一:声明式事务@Transactional
二:注解@Transactional使用
三:常用,使用该注解时,记得加异常回滚,
readOnly 属性
- readOnly :指定事务是否为只读事务,默认值为 false;为了忽略那些不需要事务的方法,比如读取数据,可以设置
read-only 为 true。
rollbackFor 属性
- rollbackFor :用于指定能够触发事务回滚的异常类型,可以指定多个异常类型。
不加异常回归的话,自定义异常失败,但不回滚;
四:使用该注解注意事项以及失效情况
总结:
三种情况失效最多是自身调用、异常被吃、异常抛出类型不对这三个了。
1、@Transactional 应用在非 public 修饰的方法上
如果Transactional注解应用在非public修饰的方法上,Transactional将会失效。
- 注意:protected、private修饰的方法上使用 @Transactional
注解,虽然事务无效,但不会有任何报错,这是我们很容犯错的一点。
2、@Transactional 注解属性 propagation 设置错误
这种失效是由于配置错误,若是错误的配置以下三种 propagation,事务将不会发生回滚。
- TransactionDefinition.PROPAGATION_SUPPORTS:如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
- TransactionDefinition.PROPAGATION_NOT_SUPPORTED:以非事务方式运行,如果当前存在事务,则把当前事务挂起。
- TransactionDefinition.PROPAGATION_NEVER:以非事务方式运行,如果当前存在事务,则抛出异常。
3、@Transactional 注解属性 rollbackFor 设置错误
rollbackFor 可以指定能够触发事务回滚的异常类型。Spring默认抛出了未检查unchecked异常(继承自
RuntimeException 的异常)或者 Error才回滚事务;其他异常不会触发回滚事务。如果在事务中抛出其他类型的异常,但却期望
Spring 能够回滚事务,就需要指定 rollbackFor属性。
4、同一个类中方法调用,导致@Transactional失效
对同一个类里面的方法调用,比如有一个类Test,它的一个方法A,A再调用本类的方法B(不论方法B是用public还是private修饰),但方法A没有声明注解事务,而B方法有。则外部调用方法A之后,方法B的事务是不会起作用的。
update方法上面没有加 @Transactional 注解,调用有 @Transactional 注解的 updateOrder 方法,updateOrder 方法上的事务不管用
public class OrderServiceImpl implements OrderService {
public void update(Order order) {
updateOrder(order);
}
@Transactional
public void updateOrder(Order order) {
// update order
}
}
5、异常被你的 catch“吃了”导致@Transactional失效
把异常吃了,然后又不抛出来,事务怎么回滚吧!
在业务方法中一般不需要catch异常,如果非要catch一定要抛出throw new RuntimeException(),或者注解中指定抛异常类型@Transactional(rollbackFor=Exception.class),否则会导致事务失效,数据commit造成数据不一致,所以有些时候try catch反倒会画蛇添足。
// @Service
public class OrderServiceImpl implements OrderService {
@Transactional
public void updateOrder(Order order) {
try {
// update order
} catch {
}
}
}
6、数据库引擎不支持事务
这种情况出现的概率并不高,事务能否生效数据库引擎是否支持事务是关键。常用的MySQL数据库默认使用支持事务的innodb引擎。一旦数据库引擎切换成不支持事务的myisam,那事务就从根本上失效了。
- 异常类型错误
这样事务也是不生效的,因为默认回滚的是:RuntimeException,如果你想触发其他异常的回滚,需要在注解上配置一下,如:
@Transactional(rollbackFor = Exception.class)
// @Service
public class OrderServiceImpl implements OrderService {
@Transactional
public void updateOrder(Order order) {
try {
// update order
} catch {
throw new Exception("更新错误");
}
}
}
- 不支持事务
Propagation.NOT_SUPPORTED: 表示不以事务运行,当前若存在事务则挂起,都主动不支持以事务方式运行了,那事务生效也是白搭!
@Service
public class OrderServiceImpl implements OrderService {
@Transactional
public void update(Order order) {
updateOrder(order);
}
@Transactional(propagation = Propagation.NOT_SUPPORTED)
public void updateOrder(Order order) {
// update order
}
}
更多推荐
所有评论(0)