8.1 题目

请简单介绍 Spring 支持的常用数据库事务传播属性和事务隔离级别

8.2 事务的传播行为

当事务方法被另一个事务方法调用时,必须指定事务应该如何传播。例如:方法可能继续在现有事务中运行,也可能开启一个新事务,并在自己的事务中运行。
事务的传播行为可以由传播属性指定。Spring 定义了 7 种类传播行为。

传播属性 描述
REQUIRED 如果有事务在运行,当前的方法就在这个事务内运行,否则,就启动一个新的事务,并在自己的事务内运行
REQUIRES_NEW 当前的方法必须启动新事务,并在它自己的事务内运行.如果有事务正在运行,应该将它挂起
SUPPORTS 如果有事务在运行,当前的方法就在这个事务内运行。否则它可以不运行在事务中。
NOT_SUPPORTED 当前的方法不应该运行在事务中。如果有运行的事务,将它挂起
MANDATORY 当前的方法必须运行在事务内部,如果没有正在运行的事务,就抛出异常
NEVER 当前的方法不应该运行在事务中。如果有运行的事务,就抛出异常
NESTED 如果有事务在运行,当前的方法就应该在这个事务的嵌套事务内运行。否则,就启动一个新的事务,并在它自己的事务内运行。

8.2.1 说明

  1. REQUIRED 传播行为

当 bookService 的 purchase() 方法被另一个事务方法 checkout() 调用时,它默认会在现有的事务内运行。这个默认的传播行为就是 REQUIRED。因此在 checkout() 方法的开始和终止边界内只有一个事务。这个事务只在 checkout() 方法结束的时候被提交,结果用户一本书都买不了。
image.png

  1. REQUIRES_NEW 传播行为

表示该方法必须启动一个新事务,并在自己的事务内运行。如果有事务在运行,就应该先挂起它。
image.png

8.3 事务的隔离级别

8.3.1 数据库事务并发问题

假设现在有两个事务:Transaction01 和 Transaction02 并发执行。

  1. 脏读
    1. Transaction01 将某条记录的 AGE 值从 20 修改为 30。
    2. Transaction02 读取了 Transaction01 更新后的值:30。
    3. Transaction01 回滚,AGE 值恢复到了20。
    4. Transaction02 读取到的 30 就是一个无效的值。
  2. 不可重复读
    1. Transaction01 读取了 AGE 值为20。
    2. Transaction02 将 AGE 值修改为30。
    3. Transaction01 再次读取 AGE 值为30,和第一次读取不一致。
  3. 幻读

    1. Transaction01 读取了 STUDENT 表中的一部分数据。
    2. Transaction02 向 STUDENT 表中插入了新的行。
    3. Transaction01 读取了 STUDENT 表时,多出了一些行。

      8.3.2 隔离级别

      数据库系统必须具有隔离并发运行各个事务的能力,使它们不会相互影响,避免各种并发问题。一个事务与其他事务隔离的程度称为隔离级别。SQL 标准中规定了多种事务隔离级别,不同隔离级别对应不同的干扰程度,隔离级别越高,数据一致性就越好,但并发性越弱。
  4. 读未提交:READ UNCOMMITTED

允许 Transaction01 读取 Transaction02 未提交的修改。

  1. 读已提交:READ COMMITTED

要求 Transaction01 只能读取 Transaction02 已提交的修改。

  1. 可重复读:REPEATABLE READ

确保 Transaction01 可以多次从一个字段中读取到相同的值,即 Transaction01 执行期间禁止其它事务对这个字段进行更新。

  1. 串行化:SERIALIZABLE

确保 Transaction01 可以多次从一个表中读取到相同的行,在 Transaction01 执行期间,禁止其它事务对这个表进行添加、更新、删除操作。可以避免任何并发问题,但性能十分低下

  1. 各个隔离级别解决并发问题的能力见下表 | | 脏读 | 不可重复读 | 幻读 | | —- | —- | —- | —- | | READ UNCOMMITTED | 有 | 有 | 有 | | READ COMMITTED | 无 | 有 | 有 | | REPEATABLE READ | 无 | 无 | 有 | | SERIALIZABLE | 无 | 无 | 无 |

  2. 各种数据库产品对事务隔离级别的支持程度 | | Oracle | MySQL | | —- | —- | —- | | READ UNCOMMITTED | √ | × | | READ COMMITTED | √(默认) | √ | | REPEATABLE READ | × | √(默认) | | SERIALIZABLE | √ | √ |

8.4 事务的属性

  • propagation:用来设置事务的传播行为
    • 事务的传播行为:一个方法运行在了一个开启了事务的方法中时,当前方法是使用原来的事务还是开启一个新的事务
    • Propagation.REQUIRED:默认值,使用原来的事务
    • Propagation.REQUIRES_NEW:将原来的事务挂起,开启一个新的事务
  • isolation:用来设置事务的隔离级别
    • Isolation.REPEATABLE_READ:可重复读,MySQL 默认的隔离级别
    • Isolation.READ_COMMITTED:读已提交,Oracle 默认的隔离级别,开发时通常使用的隔离级别