在做业务开发时,遇到了一个事务不起作用的问题。事情是这样的,方法内部的定时任务调用了一个带事务的方法,失败后事务没有回滚。查阅资料后,问题得到解决,记录下来分享给大家。
场景
我在这里模拟一个场景,大概的调用方式就如下面的代码这样。
@Override@Transactional(rollbackFor=RuntimeException.class)publicvoidinsertUser(Useruser){userMapper.insertUser(user);thrownewRuntimeException("");}/***内部调用新增方法**@paramuser*/@OverridepublicvoidinvokeInsertUser(Useruser){this.insertUser(user);}原因
AOP使用的是动态代理的机制,它会给类生成一个代理类,事务的相关操作都在代理类上完成。内部方式使用this调用方式时,使用的是实例调用,并没有通过代理类调用方法,所以会导致事务失效。
解决办法方法一引入自身bean
在类内部通过@Autowired将本身bean引入,然后通过调用自身bean,从而实现使用AOP代理操作。
注入自身bean
@Autowired@LazyprivateUserServiceservice;
修改invokeInsertUser方法
/***解决方法一在bean中将自己注入进来*@paramuser*/@OverridepublicvoidinvokeInsertUser(Useruser){this.service.insertUser(user);}方法二通过ApplicationContext引入bean
通过ApplicationContext获取bean,通过bean调用内部方法,就使用了bean的代理类。
注入ApplicationContext
@AutowiredApplicationContextapplicationContext;
修改invokeInsertUser方法
/***解决方法二通过applicationContext获取到bean*@paramuser*/@OverridepublicvoidinvokeInsertUser(Useruser){((UserService)applicationContext.getBean("userService")).invokeInsertUser(user);}方法三通过AopContext获取当前类的代理类
通过AopContext获取当前类的代理类,直接通过代理类调用方法
在引导类上添加@EnableAspectJAutoProxy(exposeProxy=true)注解
修改invokeInsertUser方法
/***解决方法三通过applicationContext获取到bean**@paramuser*/@OverridepublicvoidinvokeInsertUser(Useruser){((UserService)AopContext.currentProxy()).invokeInsertUser(user);}
以上就是内部方法调用时,事务不起作用的原因及解决办法。
如果您发现文中错误,还请多多指教。欢迎关注个人公众号,一起交流学习。
最后
打个小广告,金九银十跳槽季,平头哥给大家整理了一份较全面的Java学习资料,欢迎扫码关注微信公众号:「平头哥的技术博文」领取,祝各位升职加薪。