Spring 事务的隔离级别

简介: Spring 事务的隔离级别

Spring 事务

Spring 事务的管理方式

  1. 编程式事务:在代码中硬编码(不推荐)
  2. 声明式事务:在配置文件中配置(推荐)

声明式事务又分为2种:

  1. 基础xml的声明式事务
  2. 基于注解的声明式事务

Spring 事务的隔离级别

TransactionDefinition 接口中定义了五个表示隔离级别的常量:

  • TransactionDefinition.ISOLATION_DEFAULT使用后端数据库默认的隔离级别,MySQL默认采用repeatable_read隔离级别 Oracle 默认采用 Read_Committed 隔离级别
  • TransactionDefinition.ISOLATION_READ_UNCOMMITTED:最低隔离级别,允许读取未提交的数据,可能会导致脏读、不可重复读、幻读
  • TransactionDefinition.ISOLATION_READ_COMMITTED:语序读取已提交的数据,可以阻止脏读,但是不可重复读和幻读仍有可能发生
  • 可重复读


  • TransactionDefinition.ISOLATION_SERIALIZABLE:序列化,最高隔离级别,完全服从 ACID 的隔离级别。所有事务依次执行互不干扰,可以防止脏读、不可重复读、幻读,这将严重影响性能,通常情况下不会用到这个级别。

Spring事务中有哪几种传播行式

  • TransactionDefinition.PROPAGATION_REQUIRED(必须):
  • 如果当前存在事务,则加入该事务;如果当前没有事务,则创建一个新的事务。
  • 我们平时使用@Transaction 默认就是使用这个事务。
  • TransactionDefinition.PROPAGATION_REQIURED_NEW:
  • 创建新的事务,如果当前存在事务则将事务挂起,也就是不管外部是否有事务,都将开启新的事务,各自互不相干;
  • TransactionDefinition.PROPAGATION_NESTED(嵌套):
  • 如果当前存在事务,则创建一个事务作为当前事务的嵌套事务,如果当前不存在事务,则等价于PROPAGATION_REQUIRED事务
  • TransactionDefinition.PROPAGATION_MANDATORY:
  • 如果当前存在事务,则加入该事务;如果当前没有事务,则抛出异常。(mandatory:强制性)


若是错误的配置以下 3 种事务传播行为,事务将不会发生回滚:

  • TransactionDefinition.PROPAGATION_SUPPORTS: 如果当前存在事务,则加入该事务;如果当前没有事务,则以非事务的方式继续运行。
  • TransactionDefinition.PROPAGATION_NOT_SUPPORTED: 以非事务方式运行,如果当前存在事务,则把当前事务挂起。
  • TransactionDefinition.PROPAGATION_NEVER: 以非事务方式运行,如果当前存在事务,则抛出异常。



方法A调用方法B,

  1. A方法加了事务,则a,b在同事务中;
  2. B方法加了事务,a,b为同一类,事务失效;a,b不同类,则B有事务。

原理:

spring 在扫描bean的时候会扫描方法上是否包含@Transactional注解,如果包含,spring会为这个bean动态地生成一个子类(即代理类,proxy),代理类是继承原来那个bean的。
此时,当这个有注解的方法被调用的时候,实际上是由代理类来调用的,代理类在调用之前就会启动transaction。然而,如果这个有注解的方法是被同一个类中的其他方法调用的,那么该方法的调用并没有通过代理类,而是直接通过原来的那个bean,所以就不会启动transaction,我们看到的现象就是@Transactional注解无效。

那回到一开始的问题,我们调用的方法A不带注解,因此代理类不开事务,而是直接调用目标对象的方法。当进入目标对象的方法后,执行的上下文已经变成目标对象本身了,因为目标对象的代码是我们自己写的,和事务没有半毛钱关系,此时你再调用带注解的方法,照样没有事务,只是一个普通的方法调用而已。
简单来说,内部调用本类方法,不会再走代理了,所以B的事务不起作用。

如果AB不同类,A方法调用了其他类的方法B(方法有加事务),故B有事务。


@Transactional(rollbackFor = Exception.class)注解

@Transaction 注解作用于类上时,该类的所有 public 方法都将具有该类的属性,同时,我们也可以在方法级别使用该标注来覆盖类级别的定义。如果类或者方法加了这个注解,那么这个类里面的方法会抛出异常,就会回滚,数据库里的数据也会回滚。

@Transaction注解中如果不配置 rollbackFor 属性,那么事务只会遇到 RuntimeException 的时候才会回滚,加上rollbackFor = Exception.class,可以让事务在遇上非运行时异常时也会回滚。


目录
相关文章
|
3天前
|
Java 数据库连接 API
Spring事务管理嵌套事务详解 : 同一个类中,一个方法调用另外一个有事务的方法
Spring事务管理嵌套事务详解 : 同一个类中,一个方法调用另外一个有事务的方法
|
3天前
|
Java Spring
spring 事务控制 设置手动回滚 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
spring 事务控制 设置手动回滚 TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
|
3天前
|
XML Java 关系型数据库
面试一口气说出Spring的声明式事务@Transactional注解的6种失效场景
面试一口气说出Spring的声明式事务@Transactional注解的6种失效场景
|
27天前
|
XML Java 数据库
Spring框架第五章(声明式事务)
Spring框架第五章(声明式事务)
|
8天前
|
Java Spring
解析Spring Boot中的事务管理机制
解析Spring Boot中的事务管理机制
|
1月前
|
Java 开发者 Spring
spring事务类型,事务传递,隔离级别?
spring事务类型,事务传递,隔离级别?
|
13天前
|
XML Java 关系型数据库
Spring6(四):JUnit、事务(2)
Spring6(四):JUnit、事务(2)
8 0
|
13天前
|
Java 测试技术 数据库
Spring6(四):JUnit、事务(1)
Spring6(四):JUnit、事务(1)
13 0
|
17天前
|
Java 数据库连接 开发者
在Spring的try-catch块中手动实现事务回滚
在Spring的try-catch块中手动实现事务回滚
15 0
|
17天前
|
XML Java 数据库
Spring5系列学习文章分享---第五篇(事务概念+特性+案例+注解声明式事务管理+参数详解 )
Spring5系列学习文章分享---第五篇(事务概念+特性+案例+注解声明式事务管理+参数详解 )
13 0