MySQL中 LBCC 和 MVCC 的理解及常见问题示例
来到golang学习网的大家,相信都是编程学习爱好者,希望在这里学习数据库相关编程知识。下面本篇文章就来带大家聊聊《MySQL中 LBCC 和 MVCC 的理解及常见问题示例》,介绍一下MySQLLBCC、MVCC,希望对大家的知识积累有所帮助,助力实战开发!
1. 事务
介绍MVCC之前,先介绍下事务:事务是为了保证数据库中数据的完整性和一致性。
事务的4个基本要素:
- 原子性(Atomicity):要么同时成功,要么同时失败。(通过undo log回滚日志实现)
- 一致性(Consistency):一方扣款 xxx 元,另一方收款 xxx 元,符合事物发展的正常逻辑(通过lock锁实现)
- 隔离性(Isolation):此时有多个类似 扣款/收款 事件同时发生,每个事件之间是相互独立的(通过 lock锁 + MVCC实现)
- 持久性(Durability):不管数据库宕机或重启,数据最终都落到了磁盘上,下次加载依然可见 (通过 redo log实现)
2. MVCC初探
目的:主要是为了 提高数据库并发性能。用更好的方式去处理 读/写 冲突,做到即使有 读/写 冲突时,也能做到不加锁,非阻塞并发读。
不同隔离级别下,可能引发的问题: 脏读:并发情况下,一方事务读到了另一方事务 “已 update 但未 commit” 的数据,破坏了事务隔离性。不可重复读:并发情况下,一方事务读到了另一方事务 “已 update 或 delete ,并 commit ” 的数据,破坏了事务隔离性。幻读:并发情况下,一方事务读到了另一方事务" insert 并 commit "的数据,导致前后读取结果不一致。
MVCC中的四种事务隔离级别:

提问:V1、V2、V3在不同事务隔离级别下读取到的值分别是:

- RU-读未提交 级别:20、20、20(可能发生:脏读、不可重复读)
- RC-读已提交 级别:18、20、20(不可能发生:脏读、可能发生:不可重复度)
- RR-可重复读 级别:18、18、20 (不可能发生:脏读、不可重复读;但是因为事务A已提交,所以V3再次查询时跟事务A是没有隔离性的要求的,因此V3读取到的是20)
3. LBCC & MVCC
- LBCC(Lock-Base Concurrency Control)基于锁的并发控制;
- MVCC(Multiversion Concurrency Control)多版本并发控制;
LBCC 锁相关:

MySQL 5.5 版本之前,默认的存储引擎是MyISAM,5.5之后默认引擎是Innodb。Innodb支持事务,包括:行锁/表锁,MyISAM不支持。 意向锁 意向共享锁/读锁(表锁类型,无法手动创建),mysql 中语法: lock in share mode意向排它锁/写锁(表锁类型,无法手动创建),mysql 中语法: for update
常见问题:为什么要加入意向锁?
意向锁并不是真正用来锁定数据的,而是用来告诉你当前表中是否已经有了被 共享锁/排它锁
锁定的数据行。如果有就没必要再去加无用的表锁了,起到一个标识作用,提高加表锁的效率(相当于高铁洗手间门上方是否有人正在使用的 “指示灯”)。
记录锁(Record Lock)、间隙锁(Gap Lock)、临键锁(Next-Key Lock):
- 介绍:临键锁 = 记录锁 + 间隙锁,是 RR 可重复读-隔离级别下独有的,
- 目的:间隙锁的出现就是为了解决可重复读隔离级别下的幻读问题

问题:如图示:执行此sql语句(先开启事务):BEGIN; SELECT * FROM tbl WHERE id > 15 FOR UPDATE; ,以下两个sql语句可以执行成功吗?
MVCC底层实现详解:
快照读(实际上为读相关的操作):读取的是记录的可见版本 (有可能是历史版本),不用加锁。
简单的 SELECT 操作,属于快照读,不加锁。
SELECT * FROM user WHERE ?
当前读(实际上为写相关的操作):在事务中,update 数据前,还要去MySQL中重新读取一遍该数据对应最新版本的记录,并且 当前读 返回的记录都会加上锁,保证其他事务不会再并发修改这条记录。以下两种方式都属于当前读,需要加锁:
- 特殊读 (加锁读):
SELECT* FROM user WHERE id = xxxLOCK IN SHARE MODE; INSERT/UPDATE/DELETE等写操作。
问题:在 RR-可重复读 的默认隔离级别下,假设起始的age为18,那么Q1和Q2对应的age分别是多少呢?


- 针对 “事务B” 分析:因为存在
UPDATE写操作,触发了 当前读,所以要先去读最新提交的版本号记录(即:事务CUPDATE后提交的记录),然后事务B再去执行自己的UPDATE操作。也就是要先去读事务C提交的最新数据为19,然后事务B自身再UPDATE加1最终变为20。 - 针对 “事务A” 分析:因为事务A本身是没有任何的写操作,仅仅是
SELECT查询操作,触发 快照读。所以事务A只认准事务 BEGIN 开始之前记录的 最新最后提交的版本号,其记录值也就是初始的18。

BEGIN事务开始的时候会创建一个快照,并为对应事务分配一个事务id,即 TRX_ID- 开启事务之前最后的版本号为:up_limit_id=999,对应 age=18
- 事务B和事务C都有
UPDATE操作(当前读),所以 row_trx_id 为自身的 TRX_ID 的值,分别是1001和1002。而事务A没有UPDATE操作(快照读),所以只认准事务A在 事务开始前 最后的版本号 up_limit_id=999,其 age=18。
总结
- 事务:是为了保证数据库中数据的完整性和一致性。事务的4个特性:ACID。
- MVCC的好处:提高数据库并发性能。用更好的方式去处理 读/写 冲突,做到即使有 读/写 冲突时,也能做到不加锁,非阻塞并发读。
- MVCC四种隔离级别 :读未提交、读已提交、可重复读(MySQL默认级别)、串行化。
- MVCC事务隔离级别中,常见的三种问题:脏读、幻读、不可重复读。在RR的默认隔离级别下,单纯的
SELECT只触发 “快照读” 。而当你包含INSERT/UPDATE/DELETE等 写操作 时,这时就会触发 当前读,也就是在事务中,在相关写操作之前会再去读取一次其他事务的最后提交记录。这里的关键在于你事务中的sql是单纯的SELECT语句(快照读),还是你事务在的sql是包含了INSERT/UPDATE/DELETE等 写操作(当前读)。 - 没有建立索引或索引失效,行锁会升级为表锁,因为找不到对应行记录。所以为了避免两个事务同时修改一张表的不同记录会导致表锁的问题,建议加上索引,这样就只是行锁,而不会升级为表锁!
- 幻读的解决关键在于 间隙锁 和 临键锁(临键锁 = 记录锁 + 间隙锁) 。
最后,补充一个问题点:
如果不声明的创建主键,会有哪些危害? 比如你的id(假设int类型)没有声明为主键,并且也没有声明唯一索引(当未声明主键时,唯一索引会被取代为主键)
- 行锁升级为表锁
- 当数据量达到顶峰的时候,可能会造成“主键冲突”,int的取值范围为2^32 -1,当未声明主键时,达到最大值范围时,id会再次重新从0开使自增,这时候可能会出现覆盖之前row_id记录的情况,造成数据丢失。相反的,如果声明主键的话,那么当id达到上限时,再次insert时会报“主键冲突”错误,这时候可以将之前的int 类型的id改为big int。
- MySQL会自动声明一个“隐藏主键 row_id”,占6字节。而你自己声明int类型的主键时,只会消耗4字节。因此这是一种资源的浪费!
今天关于《MySQL中 LBCC 和 MVCC 的理解及常见问题示例》的内容就介绍到这里了,是不是学起来一目了然!想要了解更多关于mysql的内容请关注golang学习网公众号!
一文带你理解MySql中explain结果filtered
- 上一篇
- 一文带你理解MySql中explain结果filtered
- 下一篇
- MySQL占用内存过大解决方案图文详解
-
- 数据库 · MySQL | 1天前 |
- MySQL数值函数大全及使用技巧
- 117浏览 收藏
-
- 数据库 · MySQL | 2天前 |
- 三种登录MySQL方法详解
- 411浏览 收藏
-
- 数据库 · MySQL | 3天前 |
- MySQL数据备份方法与工具推荐
- 420浏览 收藏
-
- 数据库 · MySQL | 3天前 |
- MySQL数据备份方法与工具推荐
- 264浏览 收藏
-
- 数据库 · MySQL | 4天前 |
- MySQL索引的作用是什么?
- 266浏览 收藏
-
- 数据库 · MySQL | 5天前 |
- MySQL排序原理与实战应用
- 392浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQLwhere条件查询技巧
- 333浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQL常用数据类型有哪些?怎么选更合适?
- 234浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQL常用命令大全管理员必学30条
- 448浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQL高效批量插入数据方法大全
- 416浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQL性能优化技巧大全
- 225浏览 收藏
-
- 数据库 · MySQL | 1星期前 |
- MySQL数据备份4种方法保障安全
- 145浏览 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 543次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 516次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 500次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 485次学习
-
- ChatExcel酷表
- ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
- 3166次使用
-
- Any绘本
- 探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
- 3379次使用
-
- 可赞AI
- 可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
- 3408次使用
-
- 星月写作
- 星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
- 4512次使用
-
- MagicLight
- MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
- 3788次使用
-
- MySQL脏读幻读不可重复读及事务的隔离级别和MVCC、LBCC实现
- 2022-12-30 117浏览
-
- MySQL的多版本并发控制MVCC的实现
- 2022-12-28 462浏览
-
- 详解MySQL 数据库隔离级别与MVCC
- 2023-01-09 383浏览
-
- RR 事务级别下触发幻读的场景及解决方法
- 2023-02-16 365浏览
-
- MVCC
- 2023-02-24 390浏览

