阅读 59

MySQL 事务与事务隔离级别

这是我参与8月更文挑战的第23天,活动详情查看:8月更文挑战

事务

事务是由一组 SQL 语句组成的执行单元,该执行单元要么全部执行,要么全部不执行。

事务的四个属性 ACID

  • 原子性 atomicity
    指事务的不可分割性,一个事务的所有操作要么不间断地全部被执行,要么一个也没有执行。

  • 一致性 consistency
    事务必须使数据库从一个一致性状态变换到另外一个一致性状态。比如转账前总金额为 2000,转账后还是 2000

  • 隔离性 isolation
    一个事务的执行不受其他事务的干扰,即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能相互干扰。

  • 持久性 durability
    一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来的其他操作和数据库故障不应该对其有任何影响。

案例:刘备给关羽转账 100

初始金额:

  • 刘备 1000
  • 关羽 1000

假设刘备的 id1,关羽的 id2

UPDATE account SET balance = balance - 100 WHERE id = 1;
# 中间发生了意外
UPDATE account SET balance = balance + 100 WHERE id = 2;
复制代码

如果在执行完第一句 SQL 之后,由于网络故障或其它原因,导致第二句 SQL 未执行,这就会发生数据不一致。刘备的钱只剩 900,但是关羽的钱没有增加,仍然为 1000

可以利用事务的机制来避免这种情况。只需要将两条 SQL 语句放在同一个事务中:

BEGIN;
UPDATE account SET balance = balance - 100 WHERE id = 1;
UPDATE account SET balance = balance + 100 WHERE id = 2;
COMMIT;
复制代码

BEGIN 语句标记一个事务的开始,COMMIT 用于提交事务,期间如果发生故障,MySQL 会自动回滚事务,账户还原到初始状态。如果在事务执行过程中,由于业务需要,想手动回滚事务,在没有 COMMIT 之前,可以执行 ROLLBACK 来完成。

使用 BEGIN 开启一个事务后,要么提交 COMMIT,要么回滚 ROLLBACK

隔离级别

事务隔离级别:一个事务与其他事务的隔离程度。

数据库规定了多种事务隔离级别,不同隔离级别对应不同的干扰程度,隔离级别越高,数据一致性就越好,但并发性越弱。

四种事务隔离级别:

  • 读未提交 READ UNCOMMITTED:允许事务读取未被其他事务提交的变更。脏读、不可重复读和幻读的问题都会出现。
  • 读已提交 READ COMMITTED:只允许事务读取已经被其他事务提交的变更,可以避免脏读,但不可重复读和幻读问题仍然可能出现。
  • 可重复读 REPEATABLE READ:确保事务可以多次从一个字段中读取相同的值。在这个事务持续期间,禁止其他事务对这个字段进行更新,可以避免脏读和不可重复读,但幻读的问题仍然存在。
  • 可串行化 SERIALIZABLE:确保事务可以从一个表中读取相同的行。在这个事务的持续期间,禁止其他事务对该表执行插入、更新和删除操作,所有[[202012072244 数据库并发问题:脏读、不可重复读、幻读|并发问题]]都可以避免,但性能十分低下。

Oracle 支持的 2 种事务隔离级别:

  • READ COMMITTED 读已提交,这也是 Oracle 默认的事务隔离级别。
  • SERIALIZABLE 串行化。

MySQL 支持 4 种事务隔离级别,默认为 REPEATABLE READ 可重复读。

隔离级别可以解决的并发问题比较:

隔离级别脏读不可重复读幻读
读未提交×(未解决)××
读已提交√(已解决)××
可重复读×
可串行化
文章分类
后端
文章标签