史上最易懂的mysql锁 、mvvc分析

2024-06-07 09:28

本文主要是介绍史上最易懂的mysql锁 、mvvc分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1 mysql中的锁类型:

1) 表锁

表共享锁(S):表级别的读锁,表共享锁之间是兼容的。
表排他锁(X): 表级别的写锁,表排他锁和任何锁(包括表排他锁)都不兼容(不包括意向锁)。
意向排他锁(IX): 获取行排他锁之前必须获取的意向排他锁,这个锁是用了快速指示当前是否存在行排他锁,而不用在表中遍历每行数据判断当前行是否有行锁。
意向共享锁(IS): 获取行共享锁之前必须获取得意向共享锁,这个锁是用了快速指示当前是否存在意向共享锁,而不用在表中遍历每行数据判断当前行是否有行排他锁。

2) 行锁

行锁(R)主要是针对唯一索引(包括主键),行锁也分行共享锁(S)、行排他锁(X)。

3) 间隙锁

间隙锁(GAP):指锁定一个范围区间,主要用来接解决幻读问题。
插入意向锁(INSERT_INTENTION):它不是意向锁,意向锁是表锁,它是一种特殊的间隙锁,在数据插入的时候需要先获取插入意向锁。
4) 临键锁(NEXT-KEY) : 它可以看成一种组合锁,相当于行锁+间隙锁。普通的索引列(非唯一索引)就是加临键锁。间隙锁是用来锁住一个区间的,防止这个区间内插入其他数据,普通索引是可以重复的,需要锁住自身,所以还需要行锁,而唯一索引本身是有唯一性的,不能插入重复数据,只需要锁住间隙就可以了,不需要锁住自己本身。

2 具体验证

现在有一个表stu,其中的age字段有索引,现在针对age字段做锁实验。stu表中目前有如下这些数据,注意:每次写数据后都要恢复成下面的初始化数据。

CREATE TABLE `stu` (`id` int NOT NULL AUTO_INCREMENT,`name` varchar(255) DEFAULT NULL,`age` int DEFAULT NULL,PRIMARY KEY (`id`),KEY `age_index` (`age`)
) ENGINE=InnoDB AUTO_INCREMENT=125 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci;select * from stu;
+----+------+------+
| id | name | age  |
+----+------+------+
|  1 | NULL |   10 |
|  3 | 23   |   24 |
|  6 | NULL |   32 |
|  7 | 23   |   45 |
+----+------+------+

1) 间隙锁with间隙锁

间隙锁和间隙锁之间是兼容的,即可以同时同一个区间加锁,不会有锁竞争。

先在会话1中开启一个事务,更新age=24的记录,此时会获取到[10,32)这个区间的插入间隙锁,此时先不提交事务,然后在新会话2中尝试更新[10,32)区间的一条数据age=25的记录,此时会话2中sql脚本执行正常(尽管此数据不存在)、没有被阻塞。

// 会话1
COMMIT;
update stu set `name` ='hi' WHERE age=24//会话2
update stu set `name` ='hi' WHERE age=25

2) 插入意向锁with插入意向锁

插入意向锁和插入意向锁(也是一种间隙锁)之间是兼容的,只要不是唯一索引(包括主键)冲突,可以直接插入,不会有锁竞争。

先在会话1中开启一个事务,插入一条age=15的记录,此时会获取到[10,24)这个区间的插入意向锁,此时先不提交事务,然后在新会话2中尝试插入[10,24)区间的一条数据age=16的记录,此时会话2中数据插入成功、没有被阻塞。

//会话1
begin;
INSERT stu (`name`,age) values ('he',15);//会话2
INSERT stu (`name`,age) values ('he',16);

3) 间隙锁with插入意向锁

间隙锁和插入意向锁之间有锁竞争,但只有在先有间隙锁再申请插入意向锁时才会不兼容,有锁竞争;反之在先有插入意向锁再申请间隙锁是不会有锁竞争的,是兼容的。
(1)先在会话1中开启一个事务,删除记录age=25的记录(尽管此数据不存在),此时会获取到[32,45)这个区间的间隙锁,此时先不提交事务,然后在新会话2中尝试插入[32,45)区间的一条数据age=36的记录,此时会话2中数据一直被阻塞,直到会话1的事务提交。这充分证明了先有间隙锁再申请插入意向锁,锁不兼容.

//会话1
BEGIN;
DELETE from stu WHERE age=35;
//会话2
INSERT stu (`name`,age) values ('he',36); //被阻塞

(2) 其他很多技术贴都说间隙锁的锁区间是左开右闭,但我实验的结果是左闭右开。
先在会话1中开启一个事务,删除记录age=25的记录(尽管此数据不存在),此时会获取到[32,45)这个区间的间隙锁,此时先不提交事务,然后在新会话2中尝试插入[32,45)区间的一条数据age=32的记录,此时插入失败、被阻塞,然而在新会话3尝试插入另一条数据age=45的记录则成功插入、没被阻塞。按照那些技术贴的说法,应该age=45的数据插入失败被阻塞,age=32的数据插入成功,但事实却不是如此。

//会话1
BEGIN;
DELETE from stu WHERE age=35;
//会话2
INSERT stu (`name`,age) values ('he',32); //被阻塞
//会话3
INSERT stu (`name`,age) values ('he',45); //成功插入

(3)先在会话1中开启一个事务,插入一条记录age=27的记录,此时会获取到[25,32)这个区间的插入意向锁,此时先不提交事务,然后在新会话2中尝试在[25,32)区间更新的一条数据age=29的记录(尽管此数据不存在),此时会话2中sql脚本正常返回、没有被阻塞。这充分证明了先有插入意向锁再申请间隙锁,锁兼容.

//会话1
BEGIN;
INSERT stu (`name`,age) values ('he',32); //会话2
UPDATE stu set `name`='haha' WHERE age=29;//正常返回、不阻塞

3 mvvc

网上对mvvc的解释太复杂了,我这里简单说下的行为结果。
mvvc中读提交可重复读这两种隔离级别中有不同的行为。
读提交 :每次的读都是当前读,即每次都读当前最新的已提交数据。
可重复:可重复读是快照读,第一次读和当前读一样,读此时的最新的已提交数据,不同点在于之后的第2~n次读。第一次之后的第2~n次读出的数据不会再变化,这时读出的数据是第一次读的快照,它和第一次读的数据始终是一样的,不再关心第一次读之后的已提交数据。每次读数是一样的,这就是名副其实的可重复读
在可重复读隔离级别下, 普通的select 语句是快照读,而updatedeleteinsert (如果把update delete insert判断是否能写数据的过程看成读的话) select (for update)/(lock on share mode) 都是当前读。

这篇关于史上最易懂的mysql锁 、mvvc分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1038788

相关文章

python panda库从基础到高级操作分析

《pythonpanda库从基础到高级操作分析》本文介绍了Pandas库的核心功能,包括处理结构化数据的Series和DataFrame数据结构,数据读取、清洗、分组聚合、合并、时间序列分析及大数据... 目录1. Pandas 概述2. 基本操作:数据读取与查看3. 索引操作:精准定位数据4. Group

MySQL中EXISTS与IN用法使用与对比分析

《MySQL中EXISTS与IN用法使用与对比分析》在MySQL中,EXISTS和IN都用于子查询中根据另一个查询的结果来过滤主查询的记录,本文将基于工作原理、效率和应用场景进行全面对比... 目录一、基本用法详解1. IN 运算符2. EXISTS 运算符二、EXISTS 与 IN 的选择策略三、性能对比

MySQL常用字符串函数示例和场景介绍

《MySQL常用字符串函数示例和场景介绍》MySQL提供了丰富的字符串函数帮助我们高效地对字符串进行处理、转换和分析,本文我将全面且深入地介绍MySQL常用的字符串函数,并结合具体示例和场景,帮你熟练... 目录一、字符串函数概述1.1 字符串函数的作用1.2 字符串函数分类二、字符串长度与统计函数2.1

SQL Server跟踪自动统计信息更新实战指南

《SQLServer跟踪自动统计信息更新实战指南》本文详解SQLServer自动统计信息更新的跟踪方法,推荐使用扩展事件实时捕获更新操作及详细信息,同时结合系统视图快速检查统计信息状态,重点强调修... 目录SQL Server 如何跟踪自动统计信息更新:深入解析与实战指南 核心跟踪方法1️⃣ 利用系统目录

MySQL 内存使用率常用分析语句

《MySQL内存使用率常用分析语句》用户整理了MySQL内存占用过高的分析方法,涵盖操作系统层确认及数据库层bufferpool、内存模块差值、线程状态、performance_schema性能数据... 目录一、 OS层二、 DB层1. 全局情况2. 内存占js用详情最近连续遇到mysql内存占用过高导致

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

解密SQL查询语句执行的过程

《解密SQL查询语句执行的过程》文章讲解了SQL语句的执行流程,涵盖解析、优化、执行三个核心阶段,并介绍执行计划查看方法EXPLAIN,同时提出性能优化技巧如合理使用索引、避免SELECT*、JOIN... 目录1. SQL语句的基本结构2. SQL语句的执行过程3. SQL语句的执行计划4. 常见的性能优

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

SQL Server 中的 WITH (NOLOCK) 示例详解

《SQLServer中的WITH(NOLOCK)示例详解》SQLServer中的WITH(NOLOCK)是一种表提示,等同于READUNCOMMITTED隔离级别,允许查询在不获取共享锁的情... 目录SQL Server 中的 WITH (NOLOCK) 详解一、WITH (NOLOCK) 的本质二、工作

MySQL 强制使用特定索引的操作

《MySQL强制使用特定索引的操作》MySQL可通过FORCEINDEX、USEINDEX等语法强制查询使用特定索引,但优化器可能不采纳,需结合EXPLAIN分析执行计划,避免性能下降,注意版本差异... 目录1. 使用FORCE INDEX语法2. 使用USE INDEX语法3. 使用IGNORE IND