四种事务的隔离级别
SQL 标准定义的四个隔离级别为: Read Uncommitted (读未提交 ) 、 Read Committed (读已提交)、 Repeatable Read (可重复读)、 Serializable (可串行化) ,下面分别介绍。
准备: 创建一张用于测试的查询表 account表,字段自定义,并维护简单的初始化数据以供查询使用。
1、读未提交:
指一个事务可以读取到另一个未提交事务的数据。
- 打开一个客户端A,并设置当前事务模式为read uncommitted(未提交读),查询表account的初始值:
- 在客户端A的事务提交之前,打开另一个客户端B,更新表account:
- 这时,虽然客户端B的事务还没提交,但是客户端A就可以查询到B已经更新的数据。
- 一旦客户端B的事务因为某种原因回滚,所有的操作都将会被撤销,那客户端A查询到的数据其实就是脏读数据。
2、读已提交
指一个事务会读取到其他事务已提交的数据,但对于其他事务中未提交的不会再读到。
- l 打开一个客户端A,并设置当前事务模式为read committed(未提交读),查询表account的所有记录;
- l 在客户端A的事务提交之前,打开另一个客户端B,更新表account;
- l 这时,客户端B的事务还没提交,客户端A不能查询到B已经更新的数据,解决了脏读问题;
- l 客户端B的事务提交;
- l 客户端A执行与上一步相同的查询,结果 与上一步不一致,即产生了不可重复读的问题。
3、可重复读
指在同一事务中,相同的读取操作前后结果保持一致。
- l 打开一个客户端A,并设置当前事务模式为repeatable read,查询表account的所有记录
- l 在客户端A的事务提交之前,打开另一个客户端B,更新表account并提交。
- l 在客户端A查询表account的所有记录,与步骤(1)查询结果一致,没有出现不可重复读的问题。
- l 可重复读会对操作行进行加锁,对于被操作的这条行记录是不允许修改的。写到这里,应该明白的一点就是,不可重复读对应的是修改,即UPDATE操作。但是可能还会有幻读问题。因为幻读问题对应的是插入INSERT操作,而不是UPDATE操作。(这里的幻读指的是理论上存在,对于mysql等成熟的数据库,已经通过更优的算法避免了幻读)
- l 重新打开客户端B,插入一条新数据后提交;
- l 在客户端A查询表account的所有记录,没有 查出 新增数据,所以没有出现幻读
4、Serializable 序列化(串行化)
序列化是最高的事务隔离级别,在该级别下,事务串行化顺序执行,可以避免脏读、不可重复读与幻读。但是这种事务隔离级别效率低下,比较耗数据库性能,一般不使用。
总结:
1、事务隔离级别为读提交时,写数据只会锁住相应的行;
2、事务隔离级别为可重复读时,如果检索条件有索引(包括主键索引)的时候,默认加锁方式是next-key 锁;如果检索条件没有索引,更新数据时会锁住整张表。一个间隙被事务加了锁,其他事务是不能在这个间隙插入记录的,这样可以防止幻读;(避免幻读)
3、事务隔离级别为串行化时,同一时间将不会再允许其他事务型的操作。
4、隔离级别越高,越能保证数据的完整性和一致性,但是对并发性能的影响也越大。
不同的隔离级别有不同的现象,并有不同的锁和并发机制,隔离级别越高,数据库的并发性能就越差,下图为4种事隔离级别与并发性能的关系: