当前位置: 首页 > news >正文

【Mysql】记录锁、间隙锁和临键锁的区别

InnoDB通过MVCC+Next-Key Locks,解决了可重复读的事务隔离级别出现的幻读问题。

记录锁

记录锁就是为某行数据进行加锁,它封锁该行的索引记录

SELECT * FROM table WHERE id = 1 FOR UPDATE

id为1的记录行会被锁住。需要注意的的:id列必须为唯一索引列或主键列,否者上属于语句加锁就会变成临建锁。同时,查询语句必须是精准匹配(=),不能是>,<,like等,否者会退化为临建锁

执行Update操作也会加上记录锁

UPDATE table SET age = 20 WHERE id = 1

间隙锁

间隙锁是基于非唯一索引,它锁定的是一段范围内的索引记录,间隙锁锁住的是有一个区间,而不仅仅是这个区间中的每一条数据

SELECT * FROM table WHERE i BETWEEN 1 AND 10 FOR UPDATE

即所有在(1,10)区间的记录都会锁住,所有id 为 2、3、4、5、6、7、8、9 的数据行的插入会被阻塞,但是 1 和 10 两条记录行并不会被锁住

临键锁

临键锁可以理解为一种特殊的间隙锁。通过临键锁可以解决幻读问题,每个数据行上的非唯一索引列都会存在一把临键锁,当某个事务持有该数据行的临键锁时,会所主要有一段左开右闭区间的数据,需要强调的一点是,InnoDB中的行级锁是基于索引实现的,临键锁只与非唯一索引有关,在唯一索引中不存在临键锁。

假设有如下表:
MySql,InnoDB,Repeatable-Read:table(id PK, age KEY, name)

id

age

name

1

10

Lee

3

24

Soraka

5

32

Zed

7

45

Talon

该表中 age 列潜在的临键锁有:

(-∞, 10],
(10, 24],
(24, 32],
(32, 45],
(45, +∞],

事务 A 中执行如下命令:

UPDATE table SET name = Vladimir WHERE age = 24

或:

SELECT * FROM table WHERE age = 24 FOR UPDATE

之后事务B执行以下命令,则该命令会被阻塞等待:

INSERT INTO table VALUES(100, 26, 'Ezreal')

事务A在对age为24的列进行UPDATE操作的同时,也获取了(24,32]这个区间的临键锁

总结

  1. InnoDB中的行锁的实现依赖于索引,一旦某个加锁的操作没有用到索引就会退化到表锁
  2. 记录锁存在在于包括主键索引在内的唯一索引,锁定单条索引记录
  3. 间隙锁存在于非唯一索引,锁定开区间的一段间隔,它是基于临键锁实现的
  4. 临键锁存在于非唯一索引,该类型每条记录的索引都会存在这个锁,锁定一段左开右闭的索引区间

http://www.lryc.cn/news/525617.html

相关文章:

  • 神经网络|(二)sigmoid神经元函数
  • w-form-select.vue(自定义下拉框组件)(与后端字段直接相关性)
  • 【JVM】垃圾收集器详解
  • python创建一个httpServer网页上传文件到httpServer
  • 【Maui】提示消息的扩展
  • 租车骑绿岛
  • Pytorch - YOLOv11自定义资料训练
  • 微服务与docker
  • 1.23 消息队列
  • 【华为路由的arp配置】
  • 绘制决策树的尝试1
  • 概率论里的特征函数,如何用卷积定理去理解
  • Spring 是如何解决循环依赖问题
  • Linux 目录操作详解
  • Elasticsearch的经典面试题及详细解答
  • Linux-arm(1)ATF启动流程
  • C#编程:List.ForEach与foreach循环的深度对比
  • C语言文件操作:标准库与系统调用实践
  • 代码随想录 栈与队列 test 7
  • C语言练习(21)
  • 智能手机“混战”2025:谁将倒下而谁又将突围?
  • 计算机图形学:实验一 OpenGL基本绘制
  • 二分查找题目:快照数组
  • 深度学习|表示学习|卷积神经网络|参数共享是什么?|07
  • 基于相机内参推导的透视投影矩阵
  • 浅析Dubbo 原理:架构、通信与调用流程
  • 03垃圾回收篇(D3_垃圾收集器的选择及相关参数)
  • 一、引论,《组合数学(第4版)》卢开澄 卢华明
  • Vue3+TS 实现批量拖拽文件夹上传图片组件封装
  • 二叉树的所有路径(力扣257)