Springboot @Transactional注解不生效问题 最全总结

问题起因

在最近写的一个Springboot + shiro项目中,想在userService层加入事务,结果发现死活不生效
更奇怪的是,我在其他Service层加事务,100%有效!
在网上找了很多,都是些很基本的原因,后面再说怎么解决的,先把这些基本问题贴出来

基本原因

  1. @Transactional 注解只能应用到 public 可见度的方法上。 如果应用在protected、private或者 package可见度的方法上,也不会报错,不过事务设置不会起作用。
  2. 默认情况下,spring会对unchecked异常进行事务回滚;如果是checked异常则不回滚。针对这种情况,可以try catch checked异常后进行手动事务回滚。
  3. 数据库引擎要支持事务,如果是mysql,注意表要使用支持事务的引擎,比如InnoDB,如果是MyISAM,事务是不起作用的。
  4. 同一个类中, 一个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

你可能感兴趣的:(java,springboot,异常,spring,java,spring,boot,事务,注解)