本篇内容主要讲解“spring的事务传播机制详细介绍”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“spring的事务传播机制详细介绍”吧!
目录
spring的事务传播机制
背景:实习期间几次遇到事务方法,有一次本地测试时发现事务没有回滚,就把简单描述写在wx上,今天来给spring事务做个自我总结。
1、why
为什么会有事务传播机制?
场景一:
场景二:
场景三:
所以,我们需要有对应的事务传播机制来控制事务。
2、传播机制生效的条件
有了spring事务传播机制,那这种机制存在的条件呢?我们知道,spring的事务是基于aop的,确切来说,是基于JDK动态代理的AOP,这种AOP有什么特点呢? 它是基于类或者接口的,也就是说,当 @Transactional写在一个方法上时,这个方法将会被spring动态代理, 生成一个动态代理类, 对原方法进行修饰增强,但是要注意!! 原先的方法的类并没有什么不同,并没有事务,spring动态代理这个类生成的代理类才有事务,才有增强,也就是说,在同一个类里面通过this.xx()调用本类的事务方法时,事务是不会生效的,因为你调用的不是代理类。
@Transactional
@Override
public void method1() {
this.method2();
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
@Override
public void method2() {
xx
}解决方案
关键在于获取类的代理对象,而不是通过this去调用,所以以下方法都是基于这个关键点去解决的。
public class Myservice{
@Transactional
@Override
public void method1() {
((Myservice)AopContext.currentProxy()).method2();
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
@Override
public void method2() {
xx
}
}一运行,报错了,因为exposeProxy默认为false,我们要暴露代理类,就要设置为true,可以在springboot启动类上加一个注解
@EnableAspectJAutoProxy(exposeProxy = true)
public class Myservice{
@Autowired
ApplicationContext context;
Myservice service;
@PostConstruct //初始化时调用,不加也行
private void getMyservice() {
service = context.getBean(Myservice.class);
}
@Transactional
@Override
public void method1() {
service.method2();
}
@Transactional(propagation = Propagation.REQUIRES_NEW)
@Override
public void method2() {
xx
}
}第二和第三种的区别就在于,2是直接获取代理类,3是通过调用getBean间接获取代理类,总的来说,第一种是最方便的,也是最推荐的做法。
3、传播机制类型
下面的类型都是针对于被调用方法来说的,理解起来要想象成两个 service 方法的调用才可以。
PROPAGATION_REQUIRED (默认)
支持当前事务,如果当前没有事务,则新建事务
如果当前存在事务,则加入当前事务,合并成一个事务
REQUIRES_NEW (一般用在子方法需要单独事务)
NESTED
如果当前存在事务,它将会成为父级事务的一个子事务,方法结束后并没有提交,只有等父事务结束才提交
如果当前没有事务,则新建事务
如果它异常,父级可以捕获它的异常而不进行回滚,正常提交
但如果父级异常,它必然回滚,这就是和 REQUIRES_NEW 的区别
SUPPORTS
NOT_SUPPORTED
以非事务方式运行
如果当前存在事务,则把当前事务挂起
MANDATORY
NEVER
到此,相信大家对“spring的事务传播机制详细介绍”有了更深的了解,不妨来实际操作一番吧!这里是天达云网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!