Mybatis引起的全局事务失效

配置了一个全局AOP事务,TransactionAdviceConfig如下

/**
 * springboot实现事务只需要在头上加上@Transactional注解
 * @Transactional 默认只捕获RuntimeException.class
 * 对Exception异常得需要 @Transactional(rollbackFor = {Exception.class}) 捕获回滚
 * 或者使用全局AOP配置 TransactionAdviceConfig
 */
//@Transactional
/**
 * 通过AOP切面设置全局事务,拦截service包下面所有方法
 * AOP术语:通知(Advice)、连接点(Joinpoint)、切入点(Pointcut)、切面(Aspect)、目标(Target)、代理(Proxy)、织入(Weaving)
 * @author daimeng
 *
 */
@Aspect
@Configuration
public class TransactionAdviceConfig {
/*定义切点变量:拦截com包下所有类的所有方法,返回值类型任意的方法*/
//private static final String AOP_POINTCUT_EXPRESSION = "execution (* com.***.service.*.*(..))";
//..表示包含子路劲
private static final String AOP_POINTCUT_EXPRESSION = "execution (* com.dd..service..*+.*(..))";
@Autowired
    private PlatformTransactionManager transactionManager;
/**
 * springBoot事务配置
 * @return
 */
    @Bean
    public TransactionInterceptor txAdvice() {
        DefaultTransactionAttribute txAttr_REQUIRED = new DefaultTransactionAttribute();
        //如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务
        txAttr_REQUIRED.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
        //只读事物、不做更新删除等
        DefaultTransactionAttribute txAttr_REQUIRED_READONLY = new DefaultTransactionAttribute();
        txAttr_REQUIRED_READONLY.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
        txAttr_REQUIRED_READONLY.setReadOnly(true);
        
        //事务管理规则,声明具备事务管理的方法名
        NameMatchTransactionAttributeSource source = new NameMatchTransactionAttributeSource();
        source.addTransactionalMethod("add*", txAttr_REQUIRED);
        source.addTransactionalMethod("save*", txAttr_REQUIRED);
        source.addTransactionalMethod("delete*", txAttr_REQUIRED);
        source.addTransactionalMethod("update*", txAttr_REQUIRED);
        source.addTransactionalMethod("exec*", txAttr_REQUIRED);
        source.addTransactionalMethod("set*", txAttr_REQUIRED);
        source.addTransactionalMethod("get*", txAttr_REQUIRED_READONLY);
        source.addTransactionalMethod("query*", txAttr_REQUIRED_READONLY);
        source.addTransactionalMethod("find*", txAttr_REQUIRED_READONLY);
        source.addTransactionalMethod("list*", txAttr_REQUIRED_READONLY);
        source.addTransactionalMethod("count*", txAttr_REQUIRED_READONLY);
        source.addTransactionalMethod("is*", txAttr_REQUIRED_READONLY);
        return new TransactionInterceptor(transactionManager, source);
    }
    /**
     * 利用AspectJExpressionPointcut设置切面=切点+通知(写成内部bean的方式)
     * @return
     */
    @Bean
    public Advisor txAdviceAdvisor() {
    /* 声明切点的面
         * 切面(Aspect):切面就是通知和切入点的结合。通知和切入点共同定义了关于切面的全部内容——它的功能、在何时和何地完成其功能。
         * */
        AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();
        /*声明和设置需要拦截的方法,用切点语言描写*/
        pointcut.setExpression(AOP_POINTCUT_EXPRESSION);
        /*设置切面=切点pointcut+通知TxAdvice*/
        return new DefaultPointcutAdvisor(pointcut, txAdvice());
    }
}



事务类是采用AOP配置,会自动切入service包内的类,一般的serviceImpl可行,但当service类注入了Mybatis的Mapper后,全局事务就失效了,需要使用注解进行事务

比如这个service


public class UserServiceImpl implements UserService{
    @Autowired
    private UserMapper userMapper;
    @Autowired
    private SysUserRepository userRepository;
    
    @Override
    @Transactional
    public ResponseVo updateUserBscInf(User user) {
        doUpdate();
    }
}

因为UserMapper类会导致AOP事务失效,不需手动@Transactional才能启动事务,其他类这不必,比如

@Service
public class ArticleServiceImpl implements ArticleService{
    @Autowired
    private ArticleRepository articleRepository;
    
    public ArticleReadInfo addArticle(VO info) {
        doAdd();
    }
}

这个service不必要自己@Transactional也有事务。

至于为什么,根据https://blog.csdn.net/crystalqy/article/details/79667377里面的说明,是bean的装配的时候,controler取到的是没有AOP的普通bean

原因是
 Spring在使用ContextLoadListener加载applicationContext.xml或其他名称的xml文件时,能进行数据源和相关事务注解的检查,启动事务特性。如果是在
spring-mvc.xml里面的加载的话,
仅作为普通bean定义加载;如果带上事务,那么用annotation方式的事务注解和bean配置,事务会失效,要将service bean配置到xml文件中才行。 因 为spring的context是父子容器,所以会产生冲突,Controller会先进行扫描装配,而此时的Service还没有进行事务的增强处理, 得到的将是原样的Service(没有经过事务加强处理,故而没有事务处理能力) ,最后才是applicationContext.xml中的扫描配置进行事务处理。


{context}