解决@Transactional事务不回滚问题

1、事务不回滚情况 - 演示

 1.1 情况说明:

service层加了事务管理器@Transactional ,报错后,事务并没有同时回滚;

service层调用了两个dao层的方法,执行第一个dao层方法,正常往数据库插入数据。执行第二个dao层的方法报错,事务没有同时回滚。出现往考勤申请主表插入数据成功,考勤详情表插入数据失败,这样是不符合正常逻辑的;

按照正常逻辑,事务要么同时回滚,要么都执行成功;

 dao层

解决@Transactional事务不回滚问题_第1张图片

解决@Transactional事务不回滚问题_第2张图片

 service 层

解决@Transactional事务不回滚问题_第3张图片

解决@Transactional事务不回滚问题_第4张图片

 2、事务管理器失效原因

 2.1 因为 dao层、service层都加了 try{}catch,虽然在service层加了事务管理器@Transactional,事务依旧失效。

2.2 正确的处理方式是只在 Controller 加 try{}catch,dao层、service层不需要加  try{}catch,service层只需要添加事务管理器@Transactional;

2.3 这样配置后,操作数据库的方法,要么同时执行成功,要么同时执行失败。

3、事务正常生效-演示

 Controller层

解决@Transactional事务不回滚问题_第5张图片

service层、dao层

解决@Transactional事务不回滚问题_第6张图片

 解决@Transactional事务不回滚问题_第7张图片

 接口调用效果

解决@Transactional事务不回滚问题_第8张图片

 3、事务失效 - 避坑指南

3.1 如果异常被try{}catch{}了,事务就不回滚了,如果想让事务回滚必须再往外抛try{}catch{throw Exception}。

 3.2 @Transactional 注解只能加在 public 方法上,这是由 Spring AOP 的本质动态代理决定的。如果你在 protected、private 或者默认可见性的方法上使用 @Transactional 注解,这将被忽略,也不会抛出任何异常。 

3.3 @Transactional既可以作用于接口,接口方法上以及类已经类的方法上。但是Spring官方不建议接口或者接口方法上使用该注解,因为这只有在使用基于接口的代理时它才会生效。

解决@Transactional事务不回滚问题_第9张图片

你可能感兴趣的:(Spring,日积月累,Bug整理,java,spring)