- 前言
事务是保障数据正确的重要方式,事务具有原子性、一致性、隔离性和持久性四个属性。本篇文章主要说一下基于MySQL事务中的隔离性。MySQL事务的隔离性分四个级别,分别是RU(读未提交)、RC(不可重复读)、RR(可重复读)、Serializable(串行化)。不同的隔离级别,可以避免脏读、不可重复、幻读的发生,具体如下图:
non-repeatable reads:不可重复读,就是说事务A中两处读取数据,第一次读时是100,然后事务B把值改成了200,事务A再读一次,结果就发现值变了,造成A事务数据混乱。
phantom read:幻读,和不可重复读相似,也是同一个事务中多次读不一致的问题。但是不可重复读的不一致是因为它所要取的数据集被改变了,而幻读所要读的数据不一致却不是他所要读的数据改变,而是它的条件数据集改变。比如:Select id where name=”ppgogo*”,第一次读去了6个符合条件的id,第二次读时,由于事务B把第一个贴的名字由”dd”改成了“ppgogo9”,结果取出来7个数据。
创建表以及数据
开启客户端,修改隔离级别
查看隔离级别命令:
SHOW VARIABLES LIKE 'transaction_isolation';
mysql是自动提交的,需要改为手动提交
set autocommit = 0;
- 验证读未提交
1.read uncommitted
可以看到未提交的数据(脏读),举个例子:别人说的话你都相信了,但是可能他只是说说,并不实际做。
开启两个客户端 client-01,client-02 ,都设置隔离级别为读未提交。
set session TRANSACTION ISOLATION level READ UNCOMMITTED;
在客户端01修改了数据,不用提交。在client02也能查到。
client01:
begin;
update account set sal = sal + 1000 where id = '1';
client02:
dirty reads:脏读,就是说事务A未提交的数据被事务B读走,如果事务A失败回滚,将导致B所读取的数据是错误的。
** 最后client01的一定要提交。 没有提交会一直被锁,验证下面的隔离模式时会卡死
然后我们选择回滚,然后提交
ROLLBACK;
commit;
- 验证读已提交
读取提交的数据。但是,可能多次读取的数据结果不一致(不可重复读,幻读)。用读写的观点就是:读取的行数据,可以写。
将client01,client02数据隔离级别设置为读已提交
set session TRANSACTION ISOLATION level READ COMMITTED;
修改client01的数据,client02读不到client01未提交的数据。解决了脏读
-
验证可重复读
可以重复读取,但有幻读。读写观点:读取的数据行不可写,但是可以往表中新增数据。在MySQL中,其他事务新增的数据,看不到,不会产生幻读。采用多版本并发控制(MVCC)机制解决幻读问题。 -
序列化
可读,不可写。像java中的锁,写数据必须等待另一个事务结束。