可重复读隔离级别如何解决脏读、不可重复读

本文通过实验演示了可重复读隔离级别如何解决脏读和不可重复读问题,并从底层解释了其实现机制。

我们都知道可重复读隔离级别可以解决脏读、不可重复读。那么具体是如何解决的呢?
下面先通过实验来演示可重复读能够解决脏读、不可重复读问题,然后解释具体的原因。

环境搭建

1. 建立两个session连接MySQL,session1和session2

关闭session1和session2的事务自动提交。

set autocommit=0; // 关闭自动提交
select @@autocommit; // 查看当前session的自动提交状态

在这里插入图片描述
2. 将session1和session2的隔离级别设置为可重复读。

set tx_isolation='repeatable-read'; // 设置当前session的隔离级别为可重复读
select @@tx_isolation; //查看当前session的隔离级别

在这里插入图片描述
3. 随便创建一张表,我这里是user。
在这里插入图片描述
这里是user中的数据:
在这里插入图片描述

开始实验

1. 首先在session2中查询name='zz’的数据。

select * from user where name='zz';

在这里插入图片描述
2. 在session1中开启事务,然后改变name='zz’的age字段为46。

begin; // 开启事务
update user set age=46 where name='zz';  //更新数据

在这里插入图片描述
这时候session2中肯定是查不到数据的更改的,也就解决了脏读的问题:
在这里插入图片描述
3.将session1中的事务进行提交,再次到session2中查询该字段。

commit; // session1中提交事务
select * from user where name='zz'; //session2中再次查询该字段

在这里插入图片描述
这就解决了不可重复读的问题。

底层解释

1. 在InnoDB中,解决事务的隔离性(Isolation)是通过多版本并发控制(MVCC)和锁(lock)机制来实现的。
2.表(user)中的数据是有两个状态的,prepare和commit。其中prepare是事务提交之前数据的状态,而commit是事务提交之后的状态。
3.数据通过MVCC控制是具有多个版本的,其原理是:MVCC会在每行数据上增加两个字段,事务ID(DB_TRX_ID)指向前一个数据的指针(DB_ROLL_PTR),根据指针和事务ID,就可以实现行数据的多版本。
在这里插入图片描述
4.在可重复读的隔离级别下,InnoDB每次进行select查询语句,只有第一次执行select语句会产生数据快照(整张表的快照),之后执行相同的select语句,不再产生数据快照,那么下次使用相同语句查询时,访问的快照仍然是第一次产生的数据快照。那么即使当session1提交(commit)事务之后,session2访问的仍然是事务提交之前的即数据处于prepare状态的数据快照。这就解决了不可重复读的问题。

测试事务隔离级别不可重复,可按以下方法进行: ### 测试 指的是到了其他事务未提交的数据,未提交意味着这些数据可能会回滚,也就是可能最终不会存到数据库中,到了并不一定最终存在的数据,这就是。以 MySQL 数据库为例,可使用以下步骤和代码进行测试: 1. **设置隔离级别未提交**:未提交是最低隔离级别,允许到未提交的事务中的数据变更,会导致现象。 ```sql SET SESSION TRANSACTION ISOLATION LEVEL READ UNCOMMITTED; ``` 2. **开启两个事务**:分别模拟两个并发事务。 ```sql -- 事务 A START TRANSACTION; -- 事务 B START TRANSACTION; ``` 3. **事务 B 更新数据但不提交**: ```sql -- 事务 B UPDATE your_table SET column_name = 'new_value' WHERE condition; ``` 4. **事务 A 取数据**: ```sql -- 事务 A SELECT column_name FROM your_table WHERE condition; ``` 5. **事务 B 回滚**: ```sql -- 事务 B ROLLBACK; ``` 6. **事务 A 再次取数据**: ```sql -- 事务 A SELECT column_name FROM your_table WHERE condition; ``` 如果事务 A 第一次取到了事务 B 未提交更新的数据,而第二次取不到该数据,说明发生了。 ### 测试不可重复 不可重复指的是在一个事务内,最开始到的数据和事务结束前的任意时刻到的同一批数据出现不一致的情况。测试步骤和代码如下: 1. **设置隔离级别已提交**:已提交允许到已提交事务的数据,解决现象,但可能出现不可重复。 ```sql SET SESSION TRANSACTION ISOLATION LEVEL READ COMMITTED; ``` 2. **开启两个事务**: ```sql -- 事务 A START TRANSACTION; -- 事务 B START TRANSACTION; ``` 3. **事务 A 第一次取数据**: ```sql -- 事务 A SELECT column_name FROM your_table WHERE condition; ``` 4. **事务 B 更新数据并提交**: ```sql -- 事务 B UPDATE your_table SET column_name = 'new_value' WHERE condition; COMMIT; ``` 5. **事务 A 第二次取数据**: ```sql -- 事务 A SELECT column_name FROM your_table WHERE condition; ``` 如果事务 A 两次取的数据不一致,说明发生了不可重复
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值