这是我参与8月更文挑战的第23天,活动详情查看:8月更文挑战
事务
事务是由一组 SQL 语句组成的执行单元,该执行单元要么全部执行,要么全部不执行。
事务的四个属性 ACID:
-
原子性
atomicity
指事务的不可分割性,一个事务的所有操作要么不间断地全部被执行,要么一个也没有执行。 -
一致性
consistency
事务必须使数据库从一个一致性状态变换到另外一个一致性状态。比如转账前总金额为2000,转账后还是2000。 -
隔离性
isolation
一个事务的执行不受其他事务的干扰,即一个事务内部的操作及使用的数据对并发的其他事务是隔离的,并发执行的各个事务之间不能相互干扰。 -
持久性
durability
一个事务一旦被提交,它对数据库中数据的改变就是永久性的,接下来的其他操作和数据库故障不应该对其有任何影响。
案例:刘备给关羽转账 100 元
初始金额:
- 刘备
1000 - 关羽
1000
假设刘备的 id 为 1,关羽的 id 为 2。
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 可重复读。
隔离级别可以解决的并发问题比较:
| 隔离级别 | 脏读 | 不可重复读 | 幻读 |
|---|---|---|---|
| 读未提交 | ×(未解决) | × | × |
| 读已提交 | √(已解决) | × | × |
| 可重复读 | √ | √ | × |
| 可串行化 | √ | √ | √ |