SpringBoot 事务的传播机制

事务的传播机制

所谓事务的传播行为是指,如果在开始当前事务之前,一个事务上下文已经存在,此时有若干选项可以指定一个事务性方法的执行行为。

SpringBoot具有 七 种事务传播机制:

propagation值 说明
REQUIRED 如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
SUPPORTS 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
MANDATORY 如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。
REQUIRES_NEW 创建一个新的事务,如果当前存在事务,则把当前事务挂起。
这个方法会独立提交事务,不受调用者的事务影响,父级异常,它也是正常提交
NOT_SUPPORTED 以非事务方式运行,如果当前存在事务,则把当前事务挂起。
NEVER 以非事务方式运行,如果当前存在事务,则抛出异常。
NESTED 如果当前存在事务,它将会成为父级事务的一个子事务,方法结束后并没有提交,只有等父事务结束才提交
如果当前没有事务,则新建事务
如果它异常,父级可以捕获它的异常而不进行回滚,正常提交
但如果父级异常,它必然回滚,这就是和 REQUIRES_NEW 的区别

指定方法:通过使用 propagation 属性设置,例如:

@Transactional(propagation = Propagation.REQUIRED)

传播机制生效的条件

有了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() {
   ...
}

解决方案:

关键在于获取类的代理对象,而不是通过this去调用,所以,以下方法都是基于这个关键点去解决的。

1、最简单的,两个事务方法放在不同的service里面,这个比较简单,就不给例子了。(推荐)

2、AOP上下文。spring提供了AOP上下文AopContext,因此通过AopContext,可以很方便的获取到代理对象。

public class Myservice{

@Transactional
@Override
public void method1() {
   ((Myservice)AopContext.currentProxy()).method2();
}

@Transactional(propagation = Propagation.REQUIRES_NEW)
@Override
public void method2() {
   ...
}

一运行,报错了,因为exposeProxy默认为false,我们要暴露代理类,就要设置为true,可以在springboot启动类上加一个注解

@EnableAspectJAutoProxy(exposeProxy = true)

3、ApplicationContext。

public class Myservice{

   @Autowired
    ApplicationContext context;

    Myservice service;
    // Constructor >> @Autowired >> @PostConstruct
    @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() {
       ...
    }

}

第二和第三种的区别就在于,2是直接获取代理类,3是通过调用getBean间接获取代理类,总的来说,第一种是最方便的,也是最推荐的做法。

评论

  1. 博主
    iPhone Safari
    6月前
    2022-3-31 19:10:14

发送评论 编辑评论


				
|´・ω・)ノ
ヾ(≧∇≦*)ゝ
(☆ω☆)
(╯‵□′)╯︵┴─┴
 ̄﹃ ̄
(/ω\)
∠( ᐛ 」∠)_
(๑•̀ㅁ•́ฅ)
→_→
୧(๑•̀⌄•́๑)૭
٩(ˊᗜˋ*)و
(ノ°ο°)ノ
(´இ皿இ`)
⌇●﹏●⌇
(ฅ´ω`ฅ)
(╯°A°)╯︵○○○
φ( ̄∇ ̄o)
ヾ(´・ ・`。)ノ"
( ง ᵒ̌皿ᵒ̌)ง⁼³₌₃
(ó﹏ò。)
Σ(っ °Д °;)っ
( ,,´・ω・)ノ"(´っω・`。)
╮(╯▽╰)╭
o(*////▽////*)q
>﹏<
( ๑´•ω•) "(ㆆᴗㆆ)
😂
😀
😅
😊
🙂
🙃
😌
😍
😘
😜
😝
😏
😒
🙄
😳
😡
😔
😫
😱
😭
💩
👻
🙌
🖕
👍
👫
👬
👭
🌚
🌝
🙈
💊
😶
🙏
🍦
🍉
😣
Source: github.com/k4yt3x/flowerhd
颜文字
Emoji
小恐龙
花!
上一篇
下一篇