问题起因
在最近写的一个Springboot + shiro项目中,想在userService层加入事务,结果发现死活不生效
更奇怪的是,我在其他Service层加事务,100%有效!
在网上找了很多,都是些很基本的原因,后面再说怎么解决的,先把这些基本问题贴出来
基本原因
- @Transactional 注解只能应用到 public 可见度的方法上。 如果应用在protected、private或者 package可见度的方法上,也不会报错,不过事务设置不会起作用。
- 默认情况下,spring会对unchecked异常进行事务回滚;如果是checked异常则不回滚。针对这种情况,可以try catch checked异常后进行手动事务回滚。
- 数据库引擎要支持事务,如果是mysql,注意表要使用支持事务的引擎,比如InnoDB,如果是MyISAM,事务是不起作用的。
- 同一个类中, 一个no-transactional的方法去调用transactional的方法, 事务会失效。
深层原因
罪魁祸首是ShiroFilterFactoryBean
由于使用了shiro 的原因,ShiroFilterFactoryBean实现了FactoryBean,会使其优先于 使注解生效的处理器
AnnotationAwareAspectJAutoProxyCreator 加载,
默认情况下,Spring会先加载Bean的依赖,及
ShiroFilterFactoryBean 依赖于 SecurityManager 依赖于 UserRealm 依赖于 UserService
所以加载顺序为
UserService>UserRealm>SecurityManager>ShiroFilterFactoryBean 这些Bean加载完
到后面,Spring才开始处理Bean的AOP相关逻辑,但是发现这几个Bean已经被优先处理,所以会提示
XXX is not eligible for getting processed by all BeanPostProcessors (for example: not eligible for auto-proxying)
没有资格被AnnotationAwareAspectJAutoProxyCreator的BeanPostProcessor进行处理
AnnotationAwareAspectJAutoProxyCreator参考:https://www.cnblogs.com/liuyk-code/p/9886033.html
如何解决
在UserRealm里面的UserService上面加入注解@Lazy,使其加载顺序不跟ShiroFilterFactoryBean走,
使AnnotationAwareAspectJAutoProxyCreator正常代理这个Bean