理解 mvcc

mvcc 多版本并发控制,

提高数据库的并发性能 ------ 让"读"和"写"互不阻塞!

什么意思?

传统方式:读数据要加"读锁",写数据要加"写锁",读写冲突 → 等待

MVCC 方式:读走快照,写走新版本,互不影响 ✅

🎯 所以,MVCC 的诞生,首先是为了性能

然后 MVCC + "可重复读"隔离级别,确实能防止快照读下的幻读 ,

幻读就是 如果事务A 第一次查询结果为3条, 这个事务还没结束, 其他事务 插入了 数据 并且 已经提交 , 第二次查询就是 4条, 一个事务同样查询条件的 两次查询的结果不一致, 这就是幻读

在事务开始时,看到的是一个属于自己的"快照",即使别人提交了新数据,你也看不到,直到事务结束。

mvcc 的工作原理是 : 数据库(比如 MySQL InnoDB)会给每行数据加上几个"隐藏字段" :

最后修改这行的事务ID ,

指向上一个版本的"回滚指针"

当你更新一行数据时,旧数据不会立刻被覆盖,而是:

保存为一个"旧版本";

新数据作为一个"新版本";

通过指针连成一条"版本链"。

你执行查询时,数据库根据你的事务开始时间,决定你该看到哪个版本的数据。

这些"旧版本"什么时候删除?

当确认没有任何事务还需要它时,MySQL 的后台线程会清理掉这些旧版本,释放空间

当然,如果仅仅是 mvcc 是不能防止当前读 的幻读的, 并不能 防止当前读的幻读 , 比如一个事务中, 同时有两条 相同查询条件的 sql, 其中 有条语句 加了 for update , 那加了 for update 的语句走的不是快照读, 是当前读( 读取最新的) , 那两次查询的结果不一样 , 那这就是幻读了

如果 加锁, mvcc 间隙锁 , 其他事务插入不了数据了, 或者即将插入的地方在锁的范围内 (在间隙锁范围内), 那就肯定不会幻读 . 比如:

sql 复制代码
-- 事务A
BEGIN;
SELECT * FROM users WHERE age > 25 FOR UPDATE;  -- 加锁,包括间隙锁
-- 此时事务B尝试插入 age=28
-- ❌ 事务B会被阻塞,无法插入,直到事务A提交
SELECT * FROM users WHERE age > 25 FOR UPDATE;  -- 仍然只有原来的记录
COMMIT;
相关推荐
0xDevNull17 分钟前
MySQL数据冷热分离详解
后端·mysql
一江寒逸35 分钟前
零基础从入门到精通MySQL(中篇):进阶篇——吃透多表查询、事务核心与高级特性,搞定复杂业务SQL
数据库·sql·mysql
D4c-lovetrain37 分钟前
linux个人心得22 (mysql)
数据库·mysql
做个文艺程序员1 小时前
MySQL安全加固十大硬核操作
数据库·mysql·安全
MaCa .BaKa2 小时前
47-心里健康咨询平台/心理咨询系统
java·spring boot·mysql·tomcat·maven·intellij-idea·个人开发
一江寒逸2 小时前
零基础从入门到精通MySQL(上篇):筑基篇——吃透核心概念与基础操作,打通SQL入门第一关
数据库·sql·mysql
爱莉希雅&&&3 小时前
linux中MySQL数据库备份恢复的四种方法(更新中)
linux·数据库·mysql·数据库备份·mysqldumper
枕布响丸辣4 小时前
Python 操作 MySQL 数据库从入门到精通
数据库·python·mysql
Java开发追求者4 小时前
windows卸载mysql教程
mysql·mysql卸载
草莓熊Lotso5 小时前
【Linux 线程进阶】进程 vs 线程资源划分 + 线程控制全详解
java·linux·运维·服务器·数据库·c++·mysql