一、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);
}