mysql锁-这条sql加了哪些锁

2024-02-27 06:52
文章标签 sql mysql database 这条

本文主要是介绍mysql锁-这条sql加了哪些锁,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

文章目录

  • 1、 InnoDB的三种行锁
  • 2、常见的加锁语句
    • 2.1、常见隐式加锁语句
    • 2.1、常见显示加锁语句
  • 3、加锁的2条规则
  • 4、案例
    • 4.1、唯一索引等值查询
    • 4.2、唯一索引范围查询
    • 4.3、非唯一索引等值查询
    • 4.4、非唯一索引范围查询

InnoDB 存储引擎中的行锁的加锁规则。

1、 InnoDB的三种行锁

Record Lock(记录锁):锁住某一行记录
Gap Lock(间隙锁):锁住一段左开右开的区间
Next-key Lock(临键锁):锁住一段左开右闭的区间 (next-key lock 实际就是 间隙锁+记录锁)

2、常见的加锁语句

2.1、常见隐式加锁语句

1)、党见的DML语句(update、delete、insert),InnoDB 会自动给相应的记录行加写锁
2)、默认情况下对于普通 SELECT 语句,InnoDB 不会加任何锁,但是在 Serializable 隔离级别下会加行级读锁

2.1、常见显示加锁语句

1)、SELECT * FROM table_name WHERE … FOR UPDATE,加行级写锁

2)、SELECT * FROM table_name WHERE … LOCK IN SHARE MODE,加行级读锁

3、加锁的2条规则

1)查找过程中访问到的对象才会加锁

2)加锁的基本单位是 Next-key Lock

4、案例

一张order表,id为主键(唯一索引),order_id普通索引(非唯一索引),remark普通列(无索引)

idorder_idremark
104a
158b
2016c
2532d
3064e

4.1、唯一索引等值查询

唯一索引等值查询时,查询的记录是否存在,加锁的规则也会不同:
1)查询记录存在时,Next-key Lock会退化成记录锁
2)查询记录不存在时,Next-key Lock会退化成间隙锁

  • 查询记录存在
 SELECT * from t_order to2 where id = 25 for update;

结合加锁的两条核心:查找过程中访问到的对象才会加锁 + 加锁的基本单位是 Next-key Lock(左开右闭),我们可以分析出,这条语句的加锁范围是 (20, 25]

不过,由于这个唯一索引等值查询的记录 id = 25 是存在的,因此,Next-key Lock 会退化成记录锁,因此最终的加锁范围是 id = 25 这一行

可以在mysql客户端开启2个事物验证:

事物一:
start transaction; 
//锁住记录Id= 25的这条
select * from t_order to2 
where id = 25
for update
commit;
事物二:
start transaction; 
//可以插入成功
insert into t_order(id,order_id,remark )values('21','44','f')
rollback;
  • 查询的记录不存在

再来看查询的记录不存在的案例:

SELECT * from t_order to2 where id = 21;

结合加锁的两条核心:查找过程中访问到的对象才会加锁 + 加锁的基本单位是 Next-key Lock(左开右闭),我们可以分析出,这条语句的加锁范围是 (20, 25]

为什么是 (20,25] 而不是 (20, 22],因为 id = 22 的记录不存在,InnoDB 先找到 id = 20 的记录,发现不匹配,于是继续往下找,发现 id = 25,因此,id = 25 的这一行被扫描到了,所以整体的加锁范围是 (20, 25]

事物一:
start transaction; 
//锁住记录Id= 25的这条
select * from t_order to2 
where id = 25
for update
commit;
事物二:
start transaction; 
//阻塞,等待事物一提交,
insert into t_order(id,order_id,remark )values('21','44','f')
rollback;

4.2、唯一索引范围查询

唯一索引范围查询的规则和等值查询的规则一样,只有一个区别,就是唯一索引的范围查询需要一直向右遍历到第一个不满足条件的记录

select * from t_order to2 
where id >= 20 and id < 22
for update;

先来看语句查询条件的前半部分 id >= 20,因此,这条语句最开始要找的第一行是 id = 20,结合加锁的两个核心,需要加上 Next-key Lock (15,20]。又由于 id 是唯一索引,且 id = 20 的这行记录是存在的,因此会退化成记录锁,也就是只会对 id = 20 这一行加锁。

再来看语句查询条件的后半部分 id < 22,由于是范围查找,就会继续往后找第一个不满足条件的记录,也就是会找到 id = 25 这一行停下来,然后加 Next-key Lock (20, 25],重点来了,但由于 id = 25 不满足 id < 22,因此会退化成间隙锁,加锁范围变为 (20, 25)
所以,上述语句在主键 id 上的最终的加锁范围是 Record Lock id = 20 以及 Gap Lock (20, 25)

4.3、非唯一索引等值查询

非唯一索引进行等值查询的时候,根据查询的记录是否存在,加锁的规则会有所不同:
1、当查询的记录是存在的,除了会加 Next-key Lock 外,还会额外加间隙锁(规则是向下遍历到第一个不符合条件的值才能停止),也就是会加两把锁

查找记录的左区间加 Next-key Lock,右区间加 Gap lock

2、当查询的记录是不存在的,Next-key Lock 会退化成间隙锁(这个规则和唯一索引的等值查询是一样的)

  • 查询记录存在
select * from t_order to2 
where order_id =16
for update;

结合加锁的两条核心,这条语句首先会对普通索引 a 加上 Next-key Lock,范围是 (8,16]

又因为是非唯一索引等值查询,且查询的记录 order_id = 16 是存在的,所以还会加上间隙锁,规则是向下遍历到第一个不符合条件的值才能停止,因此间隙锁的范围是 (16,32)

所以,上述语句在普通索引order_id 上的最终加锁范围是 Next-key Lock (8,16] 以及 Gap Lock (16,32)。

验证,开启2个事物:

事物一:
start transaction; 
select * from t_order to2 
where order_id =16
for update;
commit;事物二:
start transaction; 
//阻塞,等待事物一提交
INSERT into t_order(id,order_id ,remark )values ('35','9','ffff')
commit;
  • 查询记录不存在
select * from t_order to2 
where order_id =18
for update;

结合加锁的两条核心,这条语句首先会对普通索引 order_id 加上 Next-key Lock,范围是 (16,32]

但是由于查询的记录order_id = 18 是不存在的,因此 Next-key Lock 会退化为间隙锁,即最终在普通索引 a 上的加锁范围是 (16,32)。

4.4、非唯一索引范围查询

范围查询需要一直向右遍历到第一个不满足条件的记录,和唯一索引范围查询不同的是,非唯一索引的范围查询并不会退化成 Record Lock 或者 Gap Lock。

start transaction; 
select * from t_order to2 
where order_id >=16 and order_id <18
for update;

先来看语句查询条件的前半部分 order_id >= 16,因此,这条语句最开始要找的第一行是 order_id = 16,结合加锁的两个核心,需要加上 Next-key Lock (8,16]。虽然非唯一索引 order_id = 16 的这行记录是存在的,但此时并不会像唯一索引那样退化成记录锁。

再来看语句查询条件的后半部分 order_id < 18,由于是范围查找,就会继续往后找第一个不满足条件的记录,也就是会找到 id = 32 这一行停下来,然后加 Next-key Lock (16, 32]。虽然 id = 32 不满足 id < 18,但此时并不会向唯一索引那样退化成间隙锁。

所以,上述语句在普通索引 a 上的最终的加锁范围是 Next-key Lock (8, 16] 和 (16, 32],也就是 (8, 32]。

这篇关于mysql锁-这条sql加了哪些锁的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

MySQL更新某个字段拼接固定字符串的实现

《MySQL更新某个字段拼接固定字符串的实现》在MySQL中,我们经常需要对数据库中的某个字段进行更新操作,本文就来介绍一下MySQL更新某个字段拼接固定字符串的实现,感兴趣的可以了解一下... 目录1. 查看字段当前值2. 更新字段拼接固定字符串3. 验证更新结果mysql更新某个字段拼接固定字符串 -

python连接本地SQL server详细图文教程

《python连接本地SQLserver详细图文教程》在数据分析领域,经常需要从数据库中获取数据进行分析和处理,下面:本文主要介绍python连接本地SQLserver的相关资料,文中通过代码... 目录一.设置本地账号1.新建用户2.开启双重验证3,开启TCP/IP本地服务二js.python连接实例1.

Spring Boot项目中结合MyBatis实现MySQL的自动主从切换功能

《SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能》:本文主要介绍SpringBoot项目中结合MyBatis实现MySQL的自动主从切换功能,本文分步骤给大家介绍的... 目录原理解析1. mysql主从复制(Master-Slave Replication)2. 读写分离3.

Ubuntu中远程连接Mysql数据库的详细图文教程

《Ubuntu中远程连接Mysql数据库的详细图文教程》Ubuntu是一个以桌面应用为主的Linux发行版操作系统,这篇文章主要为大家详细介绍了Ubuntu中远程连接Mysql数据库的详细图文教程,有... 目录1、版本2、检查有没有mysql2.1 查询是否安装了Mysql包2.2 查看Mysql版本2.

基于SpringBoot+Mybatis实现Mysql分表

《基于SpringBoot+Mybatis实现Mysql分表》这篇文章主要为大家详细介绍了基于SpringBoot+Mybatis实现Mysql分表的相关知识,文中的示例代码讲解详细,感兴趣的小伙伴可... 目录基本思路定义注解创建ThreadLocal创建拦截器业务处理基本思路1.根据创建时间字段按年进

Python3.6连接MySQL的详细步骤

《Python3.6连接MySQL的详细步骤》在现代Web开发和数据处理中,Python与数据库的交互是必不可少的一部分,MySQL作为最流行的开源关系型数据库管理系统之一,与Python的结合可以实... 目录环境准备安装python 3.6安装mysql安装pymysql库连接到MySQL建立连接执行S

MySQL双主搭建+keepalived高可用的实现

《MySQL双主搭建+keepalived高可用的实现》本文主要介绍了MySQL双主搭建+keepalived高可用的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,... 目录一、测试环境准备二、主从搭建1.创建复制用户2.创建复制关系3.开启复制,确认复制是否成功4.同

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 实践案例:修改表在数据库中,表的操作主要

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