MySQL锁—全局锁、表级锁、行级锁详解

2024-03-15 19:04

本文主要是介绍MySQL锁—全局锁、表级锁、行级锁详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

MySQL 锁

MySQL的锁按照锁的粒度可以分为全局锁、表级锁和行级锁。

一、全局锁

1. 概念

  全局锁,是对整个数据库实例加锁,加锁后整个实例处于只读状态,后续的DML、DDL语句以及已经执行更新操作的事务提交语句都将被阻塞。

2. 应用场景

  数据的一致性备份

3. 语法示例

--加全局锁
flush tables with read lock; 
/*以
主机192.168.206.131
用户名root
密码123456
数据库test
为例*/
--注意,mysqldump是MySQL提供的一个数据备份工具,不是MySQL的命令,要在命令行窗口中执行
mysqldump -h192.168.206.131 -uroot -p123456 test > test.sql 
--释放锁
unlock tables;

4. 特点

  对整个数据库加全局锁,是一个比较重的操作,存在以下问题:

  • 如果在主库上备份,那么在备份期间不能执行任何更新操作,业务基本上属于停摆状态。

  • 如果在从库上备份,那么在备份期间从库不能执行主库同步过来的二进制日志( binlog ),会导致主从延迟。

  在InnoDB引擎中,我们可以在备份时加上 --single-transaction 参数来完成不加锁的一致性数据备份。底层实现是:

  • 设置事务的隔离级别为可重复读,即REPEATABLE READ
  • start transaction with consistent snapshot
  • 完成备份操作
  • 提交事务

  在Repeatable Read隔离级别下,一致性视图是在执行start transaction with consistent snapshot时创建的。因此,即使在备份过程中有其他事务更新数据,也没有影响,从而达到了数据的一致性。

二、表级锁

  表级锁,每次操作锁住整张表。锁定粒度大,发生锁冲突的概率高,并发度低,应用在MyISAM、InnoDB、BDB等存储引擎中。
主要分为以下三类:

  • 表锁
  • 元数据锁
  • 意向锁

1. 表锁

  表锁分为表共享读锁和表独占写锁两种。

-- 表名user 加锁
lock tables user read/write;
--释放锁
unlock tables;

  其特点为:加了读锁,当前客户端可以读,写会报错,其他客户端可以读,写会阻塞。加了写锁,当前客户端可以读也可以写,其他客户端的读和写都会被阻塞。

2. 元数据锁

  元数据锁( MDL )加锁过程是系统自动控制,无需显式使用。MDL锁主要作用是维护表元数据的数据一致性,在表上有活动事务的时候,不可以对元数据进行写入操作。为了避免DML与DDL冲突,保证读写的正确性
  这里的元数据,可以简单理解为就是一张表的表结构。 也就是说,某一张表涉及到未提交的事务时,是不能够修改这张表的表结构的。
  在MySQL 5.5中引入了MDL,当对一张表进行增删改查的时候,加元数据共享锁( SHARED_READ 或 SHARED_WRITE );当对表结构进行变更操作的时候,加元数据排他锁( EXCLUSIVE )。其中元数据共享锁之间是兼容的,元数据排他锁和元数据共享锁之间是互斥的。也就是说,如果我们在当前客户端开启了事务,并对某张表进行了增删改查操作,但未提交,那么在另一个客户端中,我们同样可以进行增删改查,但一旦我们想修改表的结构,则会被阻塞。
  可以通过下面的SQL查看数据库中的元数据锁情况:

select object_type,object_schema,object_name,lock_type,lock_duration from
performance_schema.metadata_locks;

3. 意向锁

  试想这样一个场景,客户端A对某张表的某行数据进行UPDATE操作,对该行加了行锁,而客户端B想要对这张表加表锁,那么客户端B就需要遍历整张表来判定这张表有没有行锁,有哪种行锁,这就导致客户端B的效率很低。所以,为了避免DML在执行时加的行锁与表锁冲突,在InnoDB中引入了意向锁,使得表锁不用检查每行数据是否加锁,使用意向锁来减少表锁的检查。
  应用了意向锁之后,客户端A,在对某张表执行DML操作时,会对涉及的行加行锁,同时也会对该表加上意向锁。而客户端B,在对这张表加表锁的时候,会根据该表上所加的意向锁来判定是否可以成功加表锁,而不用逐行判断行锁情况了。一旦事务提交,意向共享锁和意向排他锁都会自动释放。

  • 意向共享锁( IS ):由语句select … lock in share mode添加。与表锁共享锁( read )兼容,与表锁排他锁( write )互斥。
  • 意向排他锁( IX ):由insert、update、delete、select…for update添加。与表锁共享锁( read )及排他锁( write )都互斥,意向锁之间不会互斥。

  可以通过下面的SQL查看数据库中的元数据锁情况:

select object_schema,object_name,index_name,lock_type,lock_mode,lock_data from
performance_schema.data_locks;

三、行级锁

  行级锁,每次操作锁住对应的行数据。锁定粒度小,发生锁冲突的概率低,并发度高。应用在InnoDB存储引擎中。
  InnoDB的数据是基于索引组织的,行锁是通过对索引上的索引项加锁来实现的,而不是对记录加的锁。对于行级锁,主要分为以下三类:

  • 行锁( Record Lock ):锁定单个行记录的锁,防止其他事务对此行进行update和delete。在RC、RR隔离级别下都支持。
  • 间隙锁( Gap Lock ):锁定索引记录间隙(不含该记录),确保索引记录间隙不变,防止其他事务在这个间隙进行insert,产生幻读。在RR隔离级别下支持。
  • 临键锁( Next-Key Lock ):行锁和间隙锁组合,同时锁住数据,并锁住数据前面的间隙Gap。在RR隔离级别下支持。

1. 行锁

  InnoDB实现了以下两种类型的行锁:

  • 共享锁( S ):允许一个事务去读一行,阻止其他事务获得相同数据集的排它锁。
  • 排他锁( X ):允许获取排他锁的事务更新数据,阻止其他事务获得相同数据集的共享锁和排他锁。
S(共享锁)X(排他锁)
S(共享锁)兼容互斥
X(排他锁)互斥互斥

  常见的SQL语句,在执行时,所加的行锁如下:

SQL行锁类型说明
INSERT …排他锁自动加锁
UPDATE …排他锁自动加锁
DELETE …排他锁自动加锁
SELECT …不加锁
SELECT … LOCK IN SHARE MODE共享锁需要手动在SELECT之后加LOCK IN SHARE MODE
SELECT … FOR UPDATE排他锁需要手动在SELECT之后加FOR UPDATE

  默认情况下,InnoDB在REPEATABLE READ事务隔离级别运行,InnoDB使用next-key锁进行搜索和索引扫描,以防止幻读。

  • 针对唯一索引进行检索时,对已存在的记录进行等值匹配时,将会自动优化为行锁。
  • InnoDB的行锁是针对于索引加的锁,不通过索引条件检索数据,那么InnoDB将对表中的所有记录加锁,此时就会升级为表锁。

  同样,可以通过以下SQL,查看意向锁及行锁的加锁情况:

select object_schema,object_name,index_name,lock_type,lock_mode,lock_data from
performance_schema.data_locks;

2. 间隙锁&临键锁

  默认情况下,InnoDB在REPEATABLE READ事务隔离级别运行,InnoDB使用next-key锁进行搜索和索引扫描,以防止幻读。

  • 索引上的等值查询(唯一索引),给不存在的记录加锁时, 优化为间隙锁。
  • 索引上的等值查询(非唯一普通索引),向右遍历时最后一个值不满足查询需求时,next-key锁退化为间隙锁。
  • 索引上的范围查询(唯一索引),会访问到不满足条件的第一个值为止。

下面我们模拟一下这三种场景:
  如果某张表以id为主键,只有id为3和10的数据,客户端A开启事务,对id为6的数据进行UPDATE操作,那么会在索引项3和10之间(不包含3和10)加入间隙锁,此时客户端B无法插入id在3和10之间的任何数据。

  如果某张表在age字段上建立了非唯一普通索引,只有age为21、23和28的数据,客户端A开启事务,对age为23的数据进行SELECT … LOCK IN SHARE MODE操作,那么会在对索引项23加入间隙锁,同时在索引项23和28之间加入间隙锁,此时客户端B无法插入age在21和28之间的任何数据,同时由于客户端A对索引项23加入了共享锁,客户端B也无法对索引项23再加入排他锁。

  如果某张表以id为主键,有id为3~10的数据,客户端A开启事务,对id>=6的数据进行SELECT … LOCK IN SHARE MODE操作,那么会对索引项6加入行锁,在索引项7、8、9、10加入临键锁,在supremum pseudo-record(可以理解为正无穷处)也加入临键锁。

  间隙锁的唯一目的是防止其他事务插入间隙。间隙锁可以共存,一个事务采用的间隙锁不会阻止另一个事务在同一间隙上采用间隙锁。

:行级锁的机制可以避免一部分幻读的产生,但无法完全避免。例如某张表中存在id为3和10的数据,在客户端A开启事务对id为6的数据进行SELECT …,只会对该表加入元数据共享锁,客户端B在不受影响的情况下插入了id为6的数据并提交了事务,客户端A想要插入id为6的数据则发现id为6的数据已经存在,还是出现了幻读现象。

这篇关于MySQL锁—全局锁、表级锁、行级锁详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

详解如何通过Python批量转换图片为PDF

《详解如何通过Python批量转换图片为PDF》:本文主要介绍如何基于Python+Tkinter开发的图片批量转PDF工具,可以支持批量添加图片,拖拽等操作,感兴趣的小伙伴可以参考一下... 目录1. 概述2. 功能亮点2.1 主要功能2.2 界面设计3. 使用指南3.1 运行环境3.2 使用步骤4. 核

一文详解JavaScript中的fetch方法

《一文详解JavaScript中的fetch方法》fetch函数是一个用于在JavaScript中执行HTTP请求的现代API,它提供了一种更简洁、更强大的方式来处理网络请求,:本文主要介绍Jav... 目录前言什么是 fetch 方法基本语法简单的 GET 请求示例代码解释发送 POST 请求示例代码解释

详解nginx 中location和 proxy_pass的匹配规则

《详解nginx中location和proxy_pass的匹配规则》location是Nginx中用来匹配客户端请求URI的指令,决定如何处理特定路径的请求,它定义了请求的路由规则,后续的配置(如... 目录location 的作用语法示例:location /www.chinasem.cntestproxy

CSS will-change 属性示例详解

《CSSwill-change属性示例详解》will-change是一个CSS属性,用于告诉浏览器某个元素在未来可能会发生哪些变化,本文给大家介绍CSSwill-change属性详解,感... will-change 是一个 css 属性,用于告诉浏览器某个元素在未来可能会发生哪些变化。这可以帮助浏览器优化

Python基础文件操作方法超详细讲解(详解版)

《Python基础文件操作方法超详细讲解(详解版)》文件就是操作系统为用户或应用程序提供的一个读写硬盘的虚拟单位,文件的核心操作就是读和写,:本文主要介绍Python基础文件操作方法超详细讲解的相... 目录一、文件操作1. 文件打开与关闭1.1 打开文件1.2 关闭文件2. 访问模式及说明二、文件读写1.

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

详解C++中类的大小决定因数

《详解C++中类的大小决定因数》类的大小受多个因素影响,主要包括成员变量、对齐方式、继承关系、虚函数表等,下面就来介绍一下,具有一定的参考价值,感兴趣的可以了解一下... 目录1. 非静态数据成员示例:2. 数据对齐(Padding)示例:3. 虚函数(vtable 指针)示例:4. 继承普通继承虚继承5.

前端高级CSS用法示例详解

《前端高级CSS用法示例详解》在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交互和动态效果的关键技术之一,随着前端技术的不断发展,CSS的用法也日益丰富和高级,本文将深... 前端高级css用法在前端开发中,CSS(层叠样式表)不仅是用来控制网页的外观和布局,更是实现复杂交