Spring事务失效常见的八种场景
2023-09-27 14:19:45 时间
文章目录
1. 方法自调用
第一类,
@Transactional
注解未生效情况,其实这种并不是事物失效,仅仅是注解失效,注解写了更没写一样解决方案:
- 将被调用的方法拆到单独的bean中,让切面起作用
- 自己注入自己获取代理类
@EnableAspectJAutoProxy(exposeProxy = true)
+UserService userService = (UserService) AopContext.currentProxy();
Spring事务是基于AOP的,只有当代理对象调用某个方法时,Spring的事务才会生效,而在一个方法中调用this.xxx()
方法时,由于this
并不是代理对象,所以会导致事务失效
@Transactional
public void a(){
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}
// 自调用
public void b(){
// 由于调用者并不是代理对象,所以切面失败导致事物失效
a();
}
还有一种情况
// 此时相当于`a`方法上面根本就没贴事物注解一样
@Transactional(propagation = Propagation.NEVER)
public void a(){
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}
// 调用方法也加了事物注解,此时会进行回滚,但是是由b方法的事物就行回滚的
@Transactional
public void b(){
// 由于调用者并不是代理对象,所以切面失败导致事物失效
a();
}
所以解决方案就是用代理对象调用即可
public class UserService {
@Autowired // 自己注入自己
private UserService userService;
@Transactional
public void a(){
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}
// 事物生效
public void b(){
userService.a();
}
}
或者可以通过代理上下文获取当前类的代理对象,但是这种需要我们在配置类上面设置属性@EnableAspectJAutoProxy(exposeProxy = true)
public class UserService {
@Transactional
public void a(){
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}
// 事物生效
public void b(){
// 获取当前类的代理对象
UserService userService = (UserService) AopContext.currentProxy();
userService.a();
}
}
2. 方法修饰符为private
- cglib代理是基于父子类的,如果方法私有无法在子类中获取到,无法重写,只能执行父类的方法,但是父类中没有
jdbcTemplate
,会抛出NPE
- 多态口诀:new谁调用谁的方法,谁创建使用谁的属性
public class UserService {
// cglib代理是基于父子类的,如果方法私有无法在子类中获取到,只能执行父类的方法,但是父类中没有`jdbcTemplate`,`NPE`
@Transactional
private void a(){
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}
}
3. 方法是final的
同上,代理类无法重写代理的方法
4. 单独的线程调用
当
Mybatis
或者JdbcTemplate
执行SQL时,会从ThreadLocal
中去获取数据库连接对象,如果开启事物的线程和执行SQL的线程不是同一个线程,就拿不到数据库连接对象,这样,Mybatis
或者JdbcTemplate
就会重新创建一个数据库连接用来执行SQL,如果这个事物自动事物提交autocommit
默认开启的话,就不会因为异常而进行回滚
public class UserService {
// 事物失效
@Transactional
private void a(){
new Thread(() -> {
jdbcTemplate.execute("insert into t1 values(1)");
throw new RuntimeException();
}).start();
}
}
5. Spring中没加@Configuration注解
在Spring中,由于
Mybatis
或JdbcTemplate
会从ThreadLocal中获取数据库连接,但是ThreadLocal里面存的是map,即ThreadLocal<Map<DateSource, Connection>>
的结构如果没有添加
@Configuration
注解,会导致Map中存的DateSource和Mybatis
或JdbcTemplate
对象不相等,从而拿不到数据库连接,导致自己会新建一个,从而导致事物失效注意:在SpringBoot中由于自动装箱,不会出现这种问题
6. 异常被吃掉
Spring默认只会对
RuntimeException
和Error
进行回滚
7. 类没有被Spring管理
8. 数据库没有开启事物
相关文章
- spring profile 多环境配置管理
- spring基于纯注解的声明式事务控制
- spring中事务控制的一组API
- 详解Spring中Bean的作用域与生命周期
- spring boot配置@Async
- spring jdbcTemplate 事务,各种诡异,包你醍醐灌顶!
- spring cloud 的config 介绍
- 阿里一面:Spring Bean 默认是单例的,高并发情况下,如何保证并发安全?
- Spring Boot 怎么做监控?这篇总算整明白了。。。
- Spring 事务的那些坑,都在这里了!
- Spring 官网下载zip jar
- spring事务配置
- 【Spring注解驱动开发,RabbitMQ的高级特性和消息补偿机制
- 深入浅出学习Spring框架(四):IoC和AOP的应用——事务配置
- 记一次 Spring 事务配置踩坑记
- Spring Boot(六):如何优雅的使用 Mybatis
- 请解释Spring事务传播传播行为
- JBPM4.4_jBPM4.4应用(与Spring集成&自行控制事务等)
- spring后台重定向方式
- 每日一博 - 常见的Spring事务失效&事务不回滚案例集锦(事务失效)
- Spring事务专题之十、导致 Spring 事务失效常见的几种情况
- Spring事务失效的场景
- 【Spring】注解实现IOC操作,你理解了吗?
- Spring相关面试题:谈一谈你对事务的理解?
- Java Spring 【@ContextConfiguration】java世界的那些注解
- SpringBoot2.x系列教程(四十七)Spring Boot集成WebSocket之STOMP实战