脏读、幻读、不可重复读
什么是脏读、幻读、不可重复读
- ==脏读==: 读到了其他事务还没有提交的数据
- ==不可重复读==: 对某数据进行读取过程中,有其他事务对数据进行了修改 (
UPDATE、DELETE
),导致第二次读取的结果不同 - ==幻读==: 事务在做范围查询过程中,有另外一个事务对范围内新增了记录(
INSERT
),导致范围查询的结果条数不一样
什么是脏读?
#脏读 又称无效数据的读出,是指在数据库访问中,事务T1将某一值修改,然后事务T2读取该值,此后T1因为某种原因撤销对该值的修改,这就导致了T2所读取到的数据是无效的。
脏读就是指当一个事务正在访问数据,并且对数据进行了修改,而这种修改还没有提交(commit)到数据库中,这时,另外一个事务也访问这个数据,然后使用了这个数据。因为这个数据是还没有提交的数据,那么另外一个事务读到的这个数据是脏数据,依据脏数据所做的操作可能是不正确的。
什么是不可重复读?
#不可重复读 ,是指在数据库访问中,一个事务范围内两个相同的查询却返回了不同数据。这是由于查询时系统中其他事务修改的提交而引起的。比如事务T1读取某一数据,事务T2读取并修改了该数据,T1为了对读取值进行检验而再次读取该数据,便得到了不同的结果。
一种更易理解的说法是:在一个事务内,多次读同一个数据。在这个事务还没有结束时,另一个事务也访问该同一数据。那么,在第一个事务的两次读数据之间。由于第二个事务的修改,那么第一个事务读到的数据可能不一样,这样就发生了在一个事务内两次读到的数据是不一样的,因此称为不可重复读,即原始读取不可重复。
什么是幻读 ?
#幻读
幻读是指当事务不是独立执行时发生的一种现象,例如第一个事务对一个表中的数据进行了修改,比如这种修改涉及到表中的“全部数据行”。同时,第二个事务也修改这个表中的数据,这种修改是向表中插入“一行新数据”。那么,以后就会发生操作第一个事务的用户发现表中还有没有修改的数据行,就好象发生了幻觉一样.一般解决幻读的方法是增加范围锁RangeS,锁定检锁范围为只读,这样就避免了幻读。
幻读是不可重复读的一种特殊场景:当事务没有获取范围锁的情况下执行SELECT … WHERE操作可能会发生幻读。
以下是脏读、不可重复读和幻读的详细解释及相应的示例,以帮助理解这些概念在实际应用中的表现。
脏读
定义:脏读发生在一个事务读取到另一个事务尚未提交的数据。当第二个事务回滚时,第一个事务所读取的数据将是无效的。
示例:
事务A开始并插入一条记录,但尚未提交:
1
2
3-- 事务A
START TRANSACTION;
INSERT INTO accounts (id, balance) VALUES (1, 100);事务B读取了事务A插入但未提交的数据:
1
2-- 事务B
SELECT * FROM accounts WHERE id = 1; -- 读取到 balance = 100事务A回滚:
1
2-- 事务A
ROLLBACK;事务B依然认为存在这条记录,但实际上它并没有提交,导致了脏读。
不可重复读
定义:不可重复读发生在一个事务在读取某数据后,另一个事务对该数据进行了修改(如UPDATE或DELETE),导致第一次读取和第二次读取的结果不同。
示例:
事务A开始并读取账户的余额:
1
2
3-- 事务A
START TRANSACTION;
SELECT balance FROM accounts WHERE id = 1; -- 假设读取到 balance = 100事务B对同一记录进行了更新:
1
2
3
4-- 事务B
START TRANSACTION;
UPDATE accounts SET balance = 150 WHERE id = 1;
COMMIT;事务A再次读取账户余额:
1
2-- 事务A
SELECT balance FROM accounts WHERE id = 1; -- 现在读取到 balance = 150事务A发现两次读取的结果不同,导致不可重复读。
幻读
定义:幻读发生在一个事务在进行范围查询时,另一个事务对范围内新增了记录,导致第一次和第二次查询的结果条数不一致。
示例:
事务A开始并进行范围查询:
1
2
3-- 事务A
START TRANSACTION;
SELECT COUNT(*) FROM accounts WHERE balance > 100; -- 假设当前结果为 0事务B插入了一条新记录:
1
2
3
4-- 事务B
START TRANSACTION;
INSERT INTO accounts (id, balance) VALUES (2, 150);
COMMIT;事务A再次进行相同的范围查询:
1
2-- 事务A
SELECT COUNT(*) FROM accounts WHERE balance > 100; -- 现在结果为 1事务A发现两次查询的结果条数不同,导致了幻读。
总结
脏读、不可重复读和幻读是数据库事务管理中的重要概念,分别描述了不同类型的数据一致性问题。理解这些概念对于设计和实现事务管理策略至关重要,以确保数据的完整性和一致性。
- Title: 脏读、幻读、不可重复读
- Author: cccs7
- Created at : 2025-03-23 16:56:00
- Updated at : 2025-04-09 13:27:24
- Link: https://cs7eric.github.io/2025/03/23/脏读、幻读、不可重复读/
- License: This work is licensed under CC BY-NC-SA 4.0.