MySQL数据库——常见的几种锁分类

详细介绍MySQL的几种常见锁分类,如:表级锁、行级锁、页面锁、悲观锁、乐观锁、共享锁、排他锁、Gap-锁等。

在这里插入图片描述

文章目录

      • 按锁粒度分
        • 表级锁
        • 行级锁
        • 页面锁
        • 锁与索引关系
      • 按加锁机制分【逻辑上的锁】
        • 悲观锁
        • 乐观锁
          • 版本号机制
          • CAS(Compare and Swap)
          • 区别
      • 按兼容性分
        • 共享锁(读锁)
        • 排他锁(写锁)
      • 按可见性划分
        • 隐式锁
        • 显式锁
      • 按锁模式划分
        • 记录锁(Record Lock)
        • Gap 锁(Gap Lock)
        • Next-Key 锁(Next-Key Lock)
        • 意向锁(Intention Lock)
        • 插入意向锁(Insert Intention Lock)
        • 总结

按锁粒度分

表级锁
  • 开销小,加锁快,不会出现死锁,锁粒度大(整张表)。
  • 并发度低,发生锁竞争概率大,适合查询。
行级锁
  • 开销大,加锁慢,会出现死锁,锁粒度最小(一行数据)。
  • 并发度高,发生锁竞争概率小,适合并发写,事务控制。
页面锁
  • 开销、加锁速度、锁粒度、并发度都介于表级锁和行级锁之间,会出现死锁。

总结:很难说哪种锁更好,只能根据具体应用程序的特点选择合适的锁。

  • 对于查询远大于修改的场景,表级锁是合适的,因为锁粒度大但管理简单。
  • 对于并发查询并发更新少量数据的应用,行级锁是更合适的选择,它提供更高的并发性和灵活性。

锁与索引关系

行级锁并不是直接对记录行加锁,而是对行对应的索引加锁

  • 如果 SQL 语句操作了主键索引,MySQL 会锁定这条主键索引。
  • 如果 SQL 语句操作了非主键索引,MySQL 会先锁定该非主键索引,再锁定相关的主键索引。
  • 在 InnoDB 中,如果 SQL 语句不涉及索引,则会通过隐藏的聚簇索引对记录加锁。
  • 对聚簇索引加锁,实际效果和表锁一样,因为找到某一条记录就得扫描全表,锁定表的行为和表锁一致。

按加锁机制分【逻辑上的锁】

悲观锁
  • 悲观锁思想认为并发问题极易发生,因此每次操作时,无论读写,都会先对记录加锁,以防止其他线程对数据进行修改。
  • 实现方式:数据库的行锁、读锁和写锁。

例如,使用 select...for update

select * from User where name='jay' for update;

注意:如果没有索引/主键,悲观锁会是表锁;否则是行锁。

以上 SQL 会锁定 User 表中所有符合检索条件(name='jay')的记录,在本次事务提交之前,其他线程无法修改这些记录。

乐观锁
  • 乐观锁认为多个线程操作不会频繁冲突,有线程过来,先放过去修改,如果看到别的线程没修改过,就可以修改成功,如果别的线程修改过,就修改失败或者重试。
    • 如果没有修改,事务成功;如果其他线程修改过,操作失败或重试。
    • 实现方式:乐观锁一般会使用版本号机制或 CAS(Compare and Swap)算法。

步骤:
乐观锁假设并发冲突的概率较低,因此在数据读取时不会加锁。每次数据修改时,都会检查数据的版本号来保证数据一致性。具体步骤如下:

版本号机制
  1. 读取数据:读取数据时,获取当前数据的值和对应的版本号(version1)。
  2. 修改数据:在修改数据时,检查当前数据的版本号(version2)是否与读取时的版本号(version1)一致。
  3. 版本号一致:如果一致,说明没有其他事务修改数据,可以进行更新,并将版本号自增。
  4. 版本号不一致:如果不一致,说明数据已被其他事务修改,需要回滚或重试。
CAS(Compare and Swap)
  1. 读取数据:读取数据时,获取当前值(value1)。
  2. 修改数据:在修改数据时,检查当前值(value2)是否与读取时的值(value1)一致。
  3. 值一致:如果一致,执行更新操作,并将数据值修改为新的值(value2)。
  4. 值不一致:如果不一致,说明数据已被其他线程修改,需要回滚或重试。
区别

CAS更加底层,通常直接通过原子操作实现,而版本号机制则依赖于显式的版本号字段,并且需要依赖外部锁或同步机制来保证原子性。

  • 版本号机制依赖显式的版本号字段来跟踪数据修改。
  • CAS通过底层原子操作直接对内存中的数据进行比较和更新。

按兼容性分

共享锁(读锁)
  • 共享锁允许当前线程对共享资源加共享锁,其他线程可以读取该资源,可以继续追加共享锁,但不能修改此资源,也不能追加排他锁。
  • 语法:
    select id from t_table in share mode;
    
  • 多个共享锁可以共存,但共享锁与排他锁不能共存。
排他锁(写锁)
  • 排他锁会锁住共享资源,其他线程既不能读取此资源,也不能追加共享锁或排他锁。
  • 语法:
    update t_table set a=1;	// 数据库的增删改操作默认都会加排他锁
    select * from t_table for update;	// for update会对查询的记录加排他锁,确保其他事务无法修改这些记录,但它本身并不进行增、删、改操作。
    
  • 排他锁是独占的,不会与其他锁共存。

按可见性划分

  • 隐式锁 是数据库根据事务隔离级别和执行操作自动加的锁,不需要用户干预。
  • 显式锁 是用户显式指定的锁,通过 SQL 语句来控制。
隐式锁
  • 隐式锁:InnoDB 会根据事务的隔离级别和操作自动加锁,这些锁是在数据库内部隐式管理的,用户不需要显式指定。
  • InnoDB 使用 两阶段锁定协议(2PL)
    • 扩展阶段:事务可以获取锁但不能释放锁。
    • 收缩阶段:事务可以释放锁但不能再获取锁。
显式锁
  • 显式锁:显式锁是指由用户通过 SQL 语句明确指定的锁,如 SELECT FOR UPDATELOCK IN SHARE MODE 等,用户可以通过这些语句手动加锁。

注意:按照不同维度划分的锁,相互之间没有任何联系。例如,悲观锁可以是行锁,也可以是表锁。

以下是对 按锁模式划分 的锁的补充和详细解释:

按锁模式划分

记录锁(Record Lock)
  • 定义:记录锁是锁定单个数据行的锁类型。它是最精细的锁类型,作用于表中单独的行。使用记录锁时,MySQL 会锁定某一行数据,防止其他事务修改该行数据。

  • 适用场景:通常用于 行级锁,例如在执行 SELECT FOR UPDATE 时,当满足查询条件的行被返回时,会加上记录锁。记录锁是 InnoDB 行级锁的一部分,是 InnoDB 实现行级锁的基础。

  • 特点:当一个事务修改某一行时,其他事务无法修改这行数据,从而避免了并发修改同一行数据时的冲突。

  • 示例
    假设有一个名为 users 的表,其中包含字段 idname

    -- 假设事务 A 执行了以下查询并加上了记录锁
    START TRANSACTION;
    SELECT * FROM users WHERE id = 1 FOR UPDATE;  -- 锁定 id=1 的记录
    -- 其他事务不能修改 id=1 的记录,直到事务 A 提交或回滚
    

    在这个例子中,事务 A 对 id=1 的记录加了 记录锁,直到事务 A 提交或回滚,其他事务无法修改该记录。


Gap 锁(Gap Lock)
  • 定义:Gap 锁锁住的不是具体的数据行,而是数据行之间的间隙。它阻止其他事务在该间隙中插入新行。

  • 适用场景:常用于范围查询时,防止其他事务在当前事务的查询范围内插入新记录。举例来说,如果事务 A 查找某个范围内的数据并加了一个 Gap 锁,那么在该范围内,其他事务不能插入数据。

  • 特点:Gap 锁本身不会锁住某一行,而是锁住了两个数据之间的“空隙”,防止其他事务在这个空隙插入新数据。在 Next-Key 锁 中也会涉及到 Gap 锁。

  • 示例
    假设有一个 users 表,其中按 id 列建立了索引。

    -- 假设事务 A 执行了以下查询并加上了 Gap 锁
    START TRANSACTION;
    SELECT * FROM users WHERE id BETWEEN 10 AND 20 FOR UPDATE;
    -- 在事务 A 进行操作时,ID 在 10 到 20 之间的间隙会被锁住
    -- 其他事务无法在这个范围内插入新行
    

    在这个例子中,事务 A 查询了 id 在 10 和 20 之间的范围,并加了 Gap 锁,因此其他事务不能插入 id 在该范围内的新行。

Next-Key 锁(Next-Key Lock)
  • 定义:Next-Key 锁是 记录锁Gap 锁 的结合,它锁定某一行以及该行与下一行之间的间隙。它防止其他事务对当前记录和该记录的范围进行修改或插入。

  • 适用场景:Next-Key 锁通常用于范围查询时,尤其是在对某一范围内的数据进行查询并进行加锁的场景。它既锁住了当前行,又锁住了该行后面的间隙,以确保查询结果的稳定性。

  • 特点:Next-Key 锁比单独的记录锁更加复杂,它既能防止数据行的并发修改,也防止其他事务在查询的范围内插入新的数据。

  • 示例
    假设有一个 users 表,其中按 id 列建立了索引。

    -- 假设事务 A 执行了以下查询并加上了 Next-Key 锁
    START TRANSACTION;
    SELECT * FROM users WHERE id BETWEEN 10 AND 20 FOR UPDATE;
    -- 这个操作加了 Next-Key 锁,锁住了 id=10 和 id=20 的记录以及它们之间的间隙
    -- 事务 A 会锁住 id=10 和 id=20 的记录以及 id=10 和 id=20 之间的所有间隙
    

    在这个例子中,事务 A 查询了 id 在 10 和 20 之间的范围,并加上了 Next-Key 锁,它会锁住所有符合范围条件的记录以及该记录和下一条记录之间的间隙。

Next-Key 和 Gap 锁的区别:

  • Gap 锁:锁住的是行与行之间的间隙,而不锁定具体的记录行。它防止其他事务在这个间隙中插入新的记录。
  • Next-Key 锁:锁住的是数据行(记录锁)和它们之间的间隙(Gap 锁)。它阻止其他事务在该行或该行之间插入新记录。

行为上的区别

  • Gap 锁 防止插入新记录到指定的范围内,但不锁定已存在的记录行。
  • Next-Key 锁 不仅锁定指定的记录行,也锁定这些行之间的间隙,确保插入新记录的操作不发生。
意向锁(Intention Lock)
  • 定义意向锁是表级锁的一种特殊类型,关键作用是为更细粒度的锁提供一个标识,用于标示当前事务计划在更细粒度的资源(如行、页等)上加锁。这种锁并不会直接阻塞对具体行或页的访问,而是告诉数据库系统当前事务计划在某些数据上加锁,从而避免其他事务在相同的粒度上加锁,避免冲突。

  • 适用场景:意向锁主要用于行锁与表锁的协调。它主要用来优化锁的冲突检测机制。具体来说,当一个事务想在行级别加锁时,必须先获取意向锁。如果有意向锁,数据库会知道该事务正在尝试对某些行加锁,从而避免了与其他事务加锁的冲突。

  • 意向锁的种类

    • 意向共享锁(IS 锁):表示事务意图在某些行上加共享锁。
    • 意向排他锁(IX 锁):表示事务意图在某些行上加排他锁。
  • 示例
    假设有一个 users 表,事务 A 计划在某些行上加行锁。

    -- 假设事务 A 执行了以下操作
    START TRANSACTION;
    -- 首先加上意向排他锁
    SELECT * FROM users WHERE id=1 FOR UPDATE; -- 事务 A 计划对 id=1 行加排他锁
    -- 由于已经有意向锁存在,InnoDB 会允许事务 B 在表上加锁
    -- 但是,事务 B 不能在 id=1 上加行锁,避免冲突
    

    在这个例子中,事务 A 对 id=1 的行加上了 意向排他锁(IX 锁),表示事务 A 将要在该行加上排他锁,其他事务无法对该行加锁或修改。

插入意向锁(Insert Intention Lock)
  • 定义:插入意向锁是一种特殊的意向锁,表示事务打算在某个间隙插入一行数据插入意向锁通常与 Gap 锁配合使用,以便在插入新行之前对相应的间隙进行锁定

  • 适用场景:在执行 INSERT 操作时,如果事务打算插入的行位于某个索引的间隙中,MySQL 会加上插入意向锁。这种锁不会实际锁定某行数据,而是标示事务意图插入数据。

  • 特点:插入意向锁是为了保证插入操作的顺序和一致性,避免其他事务同时在相同的间隙中插入数据。

  • 示例
    假设有一个 users 表,其中按 id 列建立了索引。

    -- 假设事务 A 执行了以下操作
    START TRANSACTION;
    INSERT INTO users (id, name) VALUES (10, 'Alice');  -- 事务 A 准备在 id=10 的位置插入数据
    -- 这会在 id=10 的位置加上插入意向锁,表示事务 A 会在该位置插入数据
    

    在这个例子中,事务 A 打算在 id=10 的位置插入数据,MySQL 会为该位置加上 插入意向锁,这与 Gap 锁 一起确保事务 A 在插入数据之前不会有其他事务在该位置插入数据。

意向锁和插入意向区别:

  • 意向锁 是为了协调 行级锁表级锁 之间的关系,标示当前事务将要在某些行上加锁。
  • 插入意向锁 是用来标示事务计划在某个空隙插入数据,并防止其他事务在该位置插入数据。它是与并发插入操作相关的一种机制。
    • Gap 锁 更侧重于锁定数据间隙,防止数据在该位置插入。
    • 插入意向锁 用于表示一个事务有意在该位置插入数据,是一种“标志锁”,通常和 Gap 锁 一起使用。

总结
  • 记录锁:最细粒度的锁,锁定单一数据行。
  • Gap 锁:锁住数据行之间的“间隙”,防止其他事务在这个空隙插入数据。
  • Next-Key 锁:结合了记录锁和 Gap 锁,锁定数据行及其后面的间隙,常用于范围查询。
  • 意向锁:用于标示事务意图加锁的粒度(行级或页级),帮助协调不同级别的锁。
  • 插入意向锁:表示事务准备在某个间隙插入数据,通常与 Gap 锁一起使用。

这些锁模式在 MySQL 的 InnoDB 存储引擎 中用于实现精细化的锁定机制,确保数据一致性和并发性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/497360.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

数据库sql语句单表查询

简单的增删改查操作 select count(*) from user where accountadmin and password123456 select count(*) from user where account"admin" insert into user(account,password) values ("admin","777") update user set password "666&…

OpenCV和PyQt的应用

1.创建一个 PyQt 应用程序,该应用程序能够: 使用 OpenCV 加载一张图像。在 PyQt 的窗口中显示这张图像。提供四个按钮(QPushButton): 一个用于将图像转换为灰度图一个用于将图像恢复为原始彩色图一个用于将图像进行翻…

电路元件与电路基本定理

电流、电压和电功率 电流 1 定义: 带电质点的有序运动形成电流 。 单位时间内通过导体横截面的电量定义为电流强度, 简称电流,用符号 i 表示,其数学表达式为:(i单位:安培(A&#x…

win11中win加方向键失效的原因

1、可能是你把win键锁了: 解决办法:先按Fn键,再按win键 2、可能是可能是 贴靠窗口设置 中将贴靠窗口关闭了,只需要将其打开就好了

十二月第五周python

第一个程序,熟悉转换器,把加法计算器变成exe# // 1,制作加法计算器, # 输入两个数字得到相加结果并输出aint(input("输入数字:"))#int()是把输入的内容转换成整数, bint(input("输入数字:&…

pyqt和pycharm环境搭建

安装 python安装: https://www.python.org/downloads/release/python-3913/ python3.9.13 64位(记得勾选Path环境变量) pycharm安装: https://www.jetbrains.com/pycharm/download/?sectionwindows community免费版 换源: pip config se…

Lottie动画源码解析

Lottie是一个很成熟的开源动画框架,它支持直接使用从AE导出的动画文件,在不同平台均可快速使用,大大减轻了程序员的工作量,也让复杂的动画成为可能。该动画文件使用Json格式来描述内容,可以大大缩减文件的体积。在Andr…

Cadence学习笔记 16 HDMI接口布局

基于Cadence 17.4,四层板4路HDMI电路 更多Cadence学习笔记:Cadence学习笔记 1 原理图库绘制Cadence学习笔记 2 PCB封装绘制Cadence学习笔记 3 MCU主控原理图绘制Cadence学习笔记 4 单片机原理图绘制Cadence学习笔记 5 四路HDMI原理图绘制Cadence学习笔记…

微服务篇-深入了解 MinIO 文件服务器(你还在使用阿里云 0SS 对象存储图片服务?教你使用 MinIO 文件服务器:实现从部署到具体使用)

🔥博客主页: 【小扳_-CSDN博客】 ❤感谢大家点赞👍收藏⭐评论✍ 文章目录 1.0 MinIO 文件服务器概述 1.1 MinIO 使用 Docker 部署 1.2 MinIO 控制台的使用 2.0 使用 Java 操作 MinIO 3.0 使用 minioClient 对象的方法 3.1 判断桶是否存在 3.2…

logback之pattern详解以及源码分析

目录 (一)pattern关键字介绍 (二)源码分析 (一)pattern关键字介绍 %d或%date:表示日期,可配置格式化%d{yyyy-MM-dd HH:mm:ss} %r或%relative:也是日期,不过…

【期末复习】JavaEE(下)

1. MVC开发模式 1.1. 运行流程 1.2. SpringMVC 核心组件 1.3. 注解解释 2. ORM与MyBatis 2.1. ORM—对象关系映射 2.2. MyBatis 2.2.1. 创建步骤 会话是单例的,不能跨方法。(单例的原因主要是从数据安全角度出发) import org.apache.ibatis…

作业帮基于 Apache DolphinScheduler 3_0_0 的缺陷修复与优化

文|作业帮大数据团队(阮文俊、孙建业) 背 景 基于 Apache DolphinScheduler (以下简称DolphinScheduler)搭建的 UDA 任务调度平台有效支撑了公司的业务数据开发需求,处理着日均百万级别的任务量。 整个 UDA 的架构如…

电脑缺失sxs.dll文件要怎么解决?

一、文件丢失问题:以sxs.dll文件缺失为例 当你在运行某个程序时,如果系统提示“找不到sxs.dll文件”,这意味着你的系统中缺少了一个名为sxs.dll的动态链接库文件。sxs.dll文件通常与Microsoft的.NET Framework相关,是许多应用程序…

Web开发:ORM框架之使用Freesql的分表分页写法

一、自动分表(高版本可用) 特性写法 //假如是按月分表:[Table(Name "log_{yyyyMM}", AsTable "createtime2022-1-1(1 month)")]注意:①需包含log_202201这张表 ②递增规律是一个月一次,确保他们…

【数据结构与算法】单向链表

一、什么是链表 链表由一系列节点组成,每个节点都包含一个 data 域(存放数据)和一个 next 域(指向下一节点)。链表中的节点可以按照任意顺序存放在内存中,它们之间并不连续。每个节点都记录了下一个节点的地…

【ACCSS】2024年亚信安全云认证专家题库

文件包含: 亚信安全ACCSS认证2019年真题(1) 亚信安全ACCSS认证2019年真题(2) 亚信安全ACCSS认证2019年真题(3) 亚信安全ACCSS认证2020年真题(1) 亚信安全ACCSS认证2020年…

OpenCV-Python实战(10)——形态学

1、腐蚀 cv2.erode() 可以删除图像中的噪音点。 可以删除毛边。 分割图像(当图像连接的不够紧密时) 。 img cv2.erode(src*,kernel*,anchor*,iterations*,borderType*,borderValue*)img:目标图像。 src:原始图像。 kernel&…

用VBA将word文档处理成支持弹出式注释的epub文档可用的html内容

有一种epub文件,其中的注释以弹窗形式显示,如下图: 点击注释引用后,对应的注释内容会弹出在页面中显示,再次点击弹窗外的任意位置该弹窗即关闭,关闭后点击任意注释引用,对应的注释内容会弹窗显示…

Ngnix介绍、安装、实战及用法!!!

一、Nginx简介 1、Nginx概述 Nginx (“engine x”) 是一个高性能的 HTTP 和 反向代理服务器,特点是占有内存少,并发能力强,能经受高负载的考验,有报告表明能支持高达 50,000 个并发连接数 。 2、正向代理 正向代理:如果把局…

【超详细】Git的基本概念和基本使用方式

Git是程序开发中非常重要的工具,是一种分布式版本控制系统,可用于管理和追踪软件开发过程中的变化。那么关于Git的基本操作你知道吗?下面是Git的基本概念和使用方式的解释: 仓库(Repository):Gi…