可重复读(Repeatable Read)隔离级别下幻读产生的原因

幻读(Phantom Read) 是指在一个事务中,两次相同的范围查询返回了不同数量 的行,主要由于其他事务插入新行导致。

产生幻读的核心原因:

1. 快照读 vs 当前读的混合使用
sql 复制代码
-- 可重复读下,普通SELECT是快照读,基于MVCC版本链
SELECT * FROM users WHERE age > 20; -- 快照读,使用事务开始时的快照

-- 但某些操作会触发当前读
UPDATE users SET status = 1 WHERE age > 20; -- 当前读,看到最新提交的数据
SELECT * FROM users WHERE age > 20 FOR UPDATE; -- 当前读,加锁
2. MVCC的局限性
  • MVCC保证已存在的行读取一致性

  • 但无法阻止其他事务插入新的行

  • 事务开始时创建ReadView,只记录当时已存在的行版本

3. 锁机制的缺失

在标准SQL规范中,可重复读级别:

  • 只对已存在的行加锁(行锁)

  • 不对不存在的行(间隙)加锁

  • 因此其他事务可以插入满足条件的新行

实际示例:

sql 复制代码
-- 事务A
START TRANSACTION;
-- 第一次查询:返回id为1,2,3的3条记录
SELECT * FROM users WHERE id BETWEEN 1 AND 5;

-- 此时事务B插入id=4的新记录并提交
INSERT INTO users(id, name) VALUES (4, 'new_user');
COMMIT;

-- 事务A再次查询(可重复读应返回相同3条记录)
SELECT * FROM users WHERE id BETWEEN 1 AND 5; -- 仍然只看到id 1,2,3

-- 但事务A执行更新操作时
UPDATE users SET status = 1 WHERE id BETWEEN 1 AND 5;
-- 更新会作用到id=4的行(因为更新是当前读)
-- 然后事务A再次查询,就会看到4条记录 ← 这就是幻读

MySQL InnoDB的特殊处理

MySQL通过Next-Key Locking机制在可重复读级别避免了幻读:

sql 复制代码
SELECT * FROM users WHERE age > 20 FOR UPDATE;
-- InnoDB会锁住age>20的整个范围(间隙锁+行锁)
-- 其他事务无法插入age>20的新行

总结表格:

隔离级别 脏读 不可重复读 幻读
读未提交
读已提交
可重复读 可能发生
串行化

关键点

  • 幻读专指新插入的行

  • 可重复读能防止已存在行的修改,但不能防止新行的插入

  • 实际应用中可通过SELECT ... FOR UPDATE加间隙锁来避免

相关推荐
宇神城主_蒋浩宇几秒前
最简单的es理解 数据库视角看写 ES 加 java正删改查深度分页
大数据·数据库·elasticsearch
2301_8174973314 分钟前
自然语言处理(NLP)入门:使用NLTK和Spacy
jvm·数据库·python
TimberWill32 分钟前
常用sql【pgsql】——建表相关
数据库·sql
herinspace37 分钟前
管家婆分销软件中如何进行现金流量分配
运维·服务器·数据库·学习·电脑
麦聪聊数据41 分钟前
LiveOps事故零容忍:游戏行业数据库的细粒度权限管控与审计实践
运维·数据库·后端·sql
敲敲千反田1 小时前
MySQL复习
数据库·mysql
SelectDB技术团队1 小时前
上市大模型企业数据基础设施的选择:MiniMax 基于阿里云 SelectDB 版,打造全球统一AI可观测中台
数据库·数据仓库·人工智能·ai·apache
小宇的天下1 小时前
Calibre :SVRF rule file example
java·开发语言·数据库
JSU_曾是此间年少1 小时前
ubuntu安装2026最新版Mysql(截止到1月底)
数据库·mysql
weixin_462446231 小时前
Hive Metastore 使用 MySQL 8(CJ 驱动)完整配置实战(含完整 Shell 脚本)
hive·hadoop·mysql