MySQL四种事务隔离级别。。。


测试mysql版本为5.6.62


一、事务的基本要素

  1. Atomicity(原子性):在事务中的一个操作要么全部执行,要么全都不执行。
  2. Consistency(一致性):事务开始前和结束后,数据的一致性没有受到破坏。比如A给B转了钱,A转了出去,B却没有收到。
  3. Isolation(隔离性):同一时间,只允许一个事务请求同一条数据,不同的事务之间互不干扰。比如修改一条数据结束之前,另外的修改操作不能执行。
  4. Durability(持久性):事务完成后,事务的所有操作将被保存到数据库,不能回滚。

二、关于脏读、不可重复读、幻读

  1. 脏读:A修改了数据,B读取A修改后的数据,但是A回滚了数据,这时候B读到的就是脏数据。
  2. 不可重复读:A多次读取同一数据,B在A读取的过程中对数据做了修改并且commit,导致A多次读取的同一数据时结果不一致。
  3. 幻读:A将数据库中所有男女的数据用0和1来代替,但是在这时候B插入了一条男女的记录,导致A在修改完成后发现还有一条男女的记录未作修改,就像产生幻觉一样。

比较:不可重复读侧重于修改同一个条数据,而幻读侧重于新增数据,解决不可重复读只需要锁住当前修改行的数据即可,而解决幻读则需要锁住整张表。

三、MySQL的事务隔离级别

事务隔离级别脏读不可重复读幻读
读未提交(read-uncommited)
读已提交(read-commited)
可重复读(repeatable-read)
串行化(serializable)

mysql的默认事务隔离级别是可重复读
通过以下语句查询:SELECT @@tx_isolation
在这里插入图片描述

四、使用MySQL演示四种事务隔离级别

1、读未提交

(1)打开客户端A,设置事务隔离级别为读未提交,
set session transaction isolation level read uncommitted;
在这里插入图片描述
(2)打开客户端B,更新一条数据。

在这里插入图片描述
(3)这个时候,虽然客户端B并未提交数据,但是客户端A已经可以读到客户端B修改之后的数据。
在这里插入图片描述

(4)这个时候由于某些原因,客户端B的数据进行回滚,那么客户端A查到的就是脏数据。
在这里插入图片描述
(5)此时在客户端A执行更新操作, update info set age = age-1 where id =1; id为1的age居然是23,而不是25-1=24.
在这里插入图片描述
要解决这个脏读的问题,可以使用数据库隔离级别中的读已提交。

2、读已提交

(1)打开客户端A,设置事务隔离级别为读已提交,查询info表中的所有数据。
在这里插入图片描述
(2)打开客户端B,设置事务隔离级别为读已提交,更新id为1的数据,但是不进行commit。
在这里插入图片描述
(3)切换到客户端A,再次查询info表中的数据,这个时候发现查询到的id为1的数据并没有被更新,这样就解决了脏读的问题。
在这里插入图片描述(4)客户端B执行commit操作。
在这里插入图片描述
(5)客户端A再次执行查询info表中的全部数据,可以发现这时候id为1的数据已经发生了改变,这样解决了读未提交的问题。
在这里插入图片描述

3、可重复读

(1)打开客户端A,设置事务隔离级别为repeatable read(可重复读),开启事务,查询所有。
在这里插入图片描述
(2)打开客户端B,设置事务隔离级别为repeatable read,开启事务,更新id为1的数据,并进行事务提交。
在这里插入图片描述
(3)打开客户端A,进行查询,发现两次的查询结果一致,在同一事务没有出现不可重复读。
在这里插入图片描述
(4)在客户端A中更新id为1的数据,发现结果和我们预想的不一样,但是这样才保证了数据的完整性没有被破坏,因为这个时候B已经提交了更新的事务。
注:可重复读的事务隔离级别下使用了MVCC机制,select不会更新版本操作,是读的数据库快照。而insert,update,delete会更新版本号,是读当前版本
在这里插入图片描述
(5)重新在客户端B中开启一个事务,插入一条数据并且提交。
在这里插入图片描述
(6)这个时候在客户端A中查询,发现并没有查到客户端B插入的数据,这个时候如果客户端A也对数据库进行插入操作,就会出现主键重复的问题。
在这里插入图片描述
客户端B执行插入数据的操作并进行提交。
在这里插入图片描述
客户端A也进行插入数据。
在这里插入图片描述

这个时候我们在A中查询并没有出现主键为9的数据,但是插入的时候发现主键重复,这样就出现了幻读。

注:关于幻读这块不是很清楚,再看看然后更新

4、序列化

(1)打开客户端A,设置数据库隔离级别为serializable,开启事务,查询info表中的所有数据。
在这里插入图片描述
(2)打开客户端B,设置数据库隔离级别为serializable,开启事务,插入一条数据,这个时候因为客户端A的事务还没有结束,所以客户端B不能执行增删改操作。串行化锁的是整张表。
在这里插入图片描述

总结:数据库的隔离级别越高数据越安全,但同时并发性能就会越低

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值