Spring 事务隔离级别和传播行为

Spring的事务隔离级别和事务的传播行为是面试中经常考察的问题,做个简单的总结。

1.传播行为

在SpringBoot中通过Transactional的propagation属性来指定,Transactional注解的具体源码如下所示

public @interface Transactional {

@AliasFor("transactionManager")
String value() default "";

@AliasFor("value")
String transactionManager() default "";

Propagation propagation() default Propagation.REQUIRED;

Isolation isolation() default Isolation.DEFAULT;

}

可以看出,默认的值是Propagation.REQUIRED;其他的还有:

1>PROPAGATION_REQUIRED:支持当前事务,假设当前没有事务。就新建一个事务。

2>PROPAGATION_SUPPORTS:支持当前事务,假设当前没有事务,就以非事务方式运行。

3>PROPAGATION_MANDATORY:支持当前事务,假设当前没有事务,就抛出异常。 

4>PROPAGATION_REQUIRES_NEW:新建事务,假设当前存在事务。把当前事务挂起。

5>PROPAGATION_NOT_SUPPORTED:以非事务方式运行操作。假设当前存在事务,就把当前事务挂起。

6>PROPAGATION_NEVER:以非事务方式运行,假设当前存在事务,则抛出异常。

7>PROPAGATION_NESTED:如果当前存在事务,则在嵌套事务内执行。如果当前没有事务,则进行与PROPAGATION_REQUIRED类似的操作。

举个例子进行说明:

ServiceA {
       
     void methodA() {
         ServiceB.methodB();
     }
  
}

ServiceB {
       
     void methodB() {
     }
       
}

 

1>PROPAGATION_REQUIRED

如果当前执行的事务不在另外一个事务里,就新起一个事务;ServiceB和 ServiceA在同一个事务里面,ServiceB如果异常,则整个事务认为是执行失败的,即便是在A里面try catch了异常也会导致A和B都回滚;同样,即便B执行成功,A执行报错产生异常,那么A和B都会回滚的;

2>PROPAGATION_SUPPORTS
假设当前在事务中。即以事务的形式执行。假设当前不再一个事务中,那么就以非事务的形式执行;

3>PROPAGATION_MANDATORY

支持当前事务,假设当前没有事务,就抛出异常,也就是说他必须在一个父事务中去执行;否则就会抛出异常;

4>PROPAGATION_REQUIRES_NEW

新建事务,假设当前存在事务,把当前事务挂起;比如服务A的事务级别是PROPAGATION_REQUIRED,那么服务B的级别是PROPAGATION_REQUIRES_NEW;那么当运行到ServiceB.methodB的时候,ServiceA.methodA所在的事务就会挂起。ServiceB.methodB会起一个新的事务,等待ServiceB.methodB的事务完毕以后,他才继续运行;跟PROPAGATION_REQUIRED的区别是会新起一个事务,而不是使用父事务,所以是两个截然不同的事务,ServiceB的执行报错,如果被ServiceA捕获了,不会影响到ServiceA的回滚;

5>PROPAGATION_NOT_SUPPORTED

当前不支持事务,比方ServiceA.methodA的事务级别是PROPAGATION_REQUIRED 。而ServiceB.methodB的事务级别是PROPAGATION_NOT_SUPPORTED ,那么当执行到ServiceB.methodB时。ServiceA.methodA的事务挂起。而他以非事务的状态执行完,再继续ServiceA.methodA的事务。

6>PROPAGATION_NEVER

不能在事务中执行。
如果ServiceA.methodA的事务级别是PROPAGATION_REQUIRED。 而ServiceB.methodB的事务级别是PROPAGATION_NEVER ,那么ServiceB.methodB就要抛出异常了。 

7>PROPAGATION_NESTED

理解Nested的关键是savepoint。
他与PROPAGATION_REQUIRES_NEW的差别是,PROPAGATION_REQUIRES_NEW另起一个事务。将会与他的父事务相互独立。而Nested的事务和他的父事务是相依的,他的提交是要等和他的父事务一块提交的。也就是说,假设父事务最后回滚。他也要回滚的。 

 

参考文章:

https://www.cnblogs.com/zsychanpin/p/7074071.html

https://blog.csdn.net/u011450220/article/details/70225598

http://www.importnew.com/24138.html

  • 8
    点赞
  • 25
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
事务隔离级别是指多个事务并发执行时,一个事务对其他事务的可见性和影响程度的控制。Spring框架支持四个事务隔离级别: 1. 未提交读(READ UNCOMMITTED):最低级别,一个事务可以读取未提交的数据,会导致脏读,不可重复读和幻读问题的出现。 2. 提交读(READ COMMITTED):一个事务只能读取已提交的数据,可以避免脏读问题,但可能导致不可重复读和幻读问题。 3. 可重复读(REPEATABLE READ):在一个事务中多次读取同一数据时,结果保持一致,避免了不可重复读问题。但仍然可能存在幻读问题。 4. 串行化(SERIALIZABLE):最高级别,通过确保事务串行执行来避免脏读、不可重复读和幻读问题。但会降低并发性能。 传播机制是指在调用多个事务方法时,如何处理事务传播Spring框架提供七种传播行为: 1. REQUIRED:如果当前没有事务,就创建一个新事务;如果已存在事务,则加入该事务。 2. SUPPORTS:支持当前事务,如果当前没有事务,就以非事务方法执行。 3. MANDATORY:强制要求存在当前事务,如果没有事务就抛出异常。 4. REQUIRES_NEW:创建一个新事务,并暂停当前事务(如果有)。 5. NOT_SUPPORTED:以非事务方式执行操作,如果当前存在事务,则将其挂起。 6. NEVER:以非事务方式执行操作,如果当前存在事务,则抛出异常。 7. NESTED:在当前事务的控制下执行一个嵌套事务,如果不存在当前事务,则创建一个新事务。嵌套事务可以独立提交或回滚,但在外部事务提交时才会生效。 通过选择合适的事务隔离级别传播机制,可以确保事务的数据一致性、安全性和性能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值