Spring事务失效的原因及解决方案
作者:码农郭大大
一、spring方法调用导致事务失效原因
1、事务失效场景复现
1.1 背景原因
在循环中处理事务问题上,我们使用声明式事务有以下选择:
1.1.1 把@Transanal注解放在整个循环的方法上,这样整个循环都会被spring事务管理到,缺点是使用了长事务,会导致锁表问题,影响效率。
1.1.2 将每一次循环抽出一个方法,把@Transanal注解加在这个方法上。这样spring只管理了本次循环的事务,解决了长事务问题,但是有事务失效的风险。
1.2 场景模拟
@Repository public class BookDao { @Transactional(propagation = Propagation.REQUIRED) public void updateStock(int id){ String sql = "update book_stock set stock=stock-1 where id=?"; jdbcTemplate.update(sql,id); for (int i = 1 ;i>=0 ;i--) System.out.println(10/i); } public void test(){ this.updateStock(1); } }
updateStock方法上@Transanal,会将updateStock方法交给spring事务管理;
test方法上无@Transanal,test方法不会交给spring事务管理。
测试的时候调用test方法的时候,发现test内部调用的updateStock时,updateStock的事务失效了;因为test内部调用的updateStock方法只是BookDao原对象的的原方法,而不是BookDao被代理对象的方法,所以事务失效
2、解决方案
通过事务失效的原因分析可知,我们只有调用代理对象的方法才会走事务
2.1 方法一
从容器中取出代理类,调用它的代理方法。可以通过实现ApplicationContextAware接口,获取IOC容器applicationContext对象,从容器中获取BookDao代理对象
@Repository public class BookDao implements ApplicationContextAware { private ApplicationContext applicationContext; @Override public void setApplicationContext(ApplicationContext applicationContext) throws BeansException { applicationContext = applicationContext; } public void test(){ BookDao bookDao = applicationContext.getBean(BookDao.class); bookDao.updateStock(1); } @Transactional(propagation = Propagation.REQUIRED) public void updateStock(int id){ String sql = "update book_stock set stock=stock-1 where id=?"; jdbcTemplate.update(sql,id); for (int i = 1 ;i>=0 ;i--) System.out.println(10/i); } }
2.1 方法二
使用AOP暴露出来的代理对象,其本质也跟上面的一样,在启动类上加@EnableAspectJAutoProxy(exposeProxy = true),方法调用通过AopContext.currentProxy()获取代理对象
public void test(){ BookDao bookDao = ( BookDao)AopContext.currentProxy(); // BookDao bookDao = applicationContext.getBean(BookDao.class); bookDao.updateStock(1); }
到此这篇关于Spring事务失效的原因及解决方案的文章就介绍到这了,更多相关Spring事务失效内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!