史上最易懂的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

相关文章

MyBatis 动态 SQL 优化之标签的实战与技巧(常见用法)

《MyBatis动态SQL优化之标签的实战与技巧(常见用法)》本文通过详细的示例和实际应用场景,介绍了如何有效利用这些标签来优化MyBatis配置,提升开发效率,确保SQL的高效执行和安全性,感... 目录动态SQL详解一、动态SQL的核心概念1.1 什么是动态SQL?1.2 动态SQL的优点1.3 动态S

Mysql表的简单操作(基本技能)

《Mysql表的简单操作(基本技能)》在数据库中,表的操作主要包括表的创建、查看、修改、删除等,了解如何操作这些表是数据库管理和开发的基本技能,本文给大家介绍Mysql表的简单操作,感兴趣的朋友一起看... 目录3.1 创建表 3.2 查看表结构3.3 修改表3.4 实践案例:修改表在数据库中,表的操作主要

Spring事务中@Transactional注解不生效的原因分析与解决

《Spring事务中@Transactional注解不生效的原因分析与解决》在Spring框架中,@Transactional注解是管理数据库事务的核心方式,本文将深入分析事务自调用的底层原理,解释为... 目录1. 引言2. 事务自调用问题重现2.1 示例代码2.2 问题现象3. 为什么事务自调用会失效3

mysql出现ERROR 2003 (HY000): Can‘t connect to MySQL server on ‘localhost‘ (10061)的解决方法

《mysql出现ERROR2003(HY000):Can‘tconnecttoMySQLserveron‘localhost‘(10061)的解决方法》本文主要介绍了mysql出现... 目录前言:第一步:第二步:第三步:总结:前言:当你想通过命令窗口想打开mysql时候发现提http://www.cpp

MySQL大表数据的分区与分库分表的实现

《MySQL大表数据的分区与分库分表的实现》数据库的分区和分库分表是两种常用的技术方案,本文主要介绍了MySQL大表数据的分区与分库分表的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有... 目录1. mysql大表数据的分区1.1 什么是分区?1.2 分区的类型1.3 分区的优点1.4 分

MySQL错误代码2058和2059的解决办法

《MySQL错误代码2058和2059的解决办法》:本文主要介绍MySQL错误代码2058和2059的解决办法,2058和2059的错误码核心都是你用的客户端工具和mysql版本的密码插件不匹配,... 目录1. 前置理解2.报错现象3.解决办法(敲重点!!!)1. php前置理解2058和2059的错误

Mysql删除几亿条数据表中的部分数据的方法实现

《Mysql删除几亿条数据表中的部分数据的方法实现》在MySQL中删除一个大表中的数据时,需要特别注意操作的性能和对系统的影响,本文主要介绍了Mysql删除几亿条数据表中的部分数据的方法实现,具有一定... 目录1、需求2、方案1. 使用 DELETE 语句分批删除2. 使用 INPLACE ALTER T

MySQL INSERT语句实现当记录不存在时插入的几种方法

《MySQLINSERT语句实现当记录不存在时插入的几种方法》MySQL的INSERT语句是用于向数据库表中插入新记录的关键命令,下面:本文主要介绍MySQLINSERT语句实现当记录不存在时... 目录使用 INSERT IGNORE使用 ON DUPLICATE KEY UPDATE使用 REPLACE

MySQL Workbench 安装教程(保姆级)

《MySQLWorkbench安装教程(保姆级)》MySQLWorkbench是一款强大的数据库设计和管理工具,本文主要介绍了MySQLWorkbench安装教程,文中通过图文介绍的非常详细,对大... 目录前言:详细步骤:一、检查安装的数据库版本二、在官网下载对应的mysql Workbench版本,要是

mysql数据库重置表主键id的实现

《mysql数据库重置表主键id的实现》在我们的开发过程中,难免在做测试的时候会生成一些杂乱无章的SQL主键数据,本文主要介绍了mysql数据库重置表主键id的实现,具有一定的参考价值,感兴趣的可以了... 目录关键语法演示案例在我们的开发过程中,难免在做测试的时候会生成一些杂乱无章的SQL主键数据,当我们