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中的扫描配置进行事务处理。