MySQL 锁机制全解析从锁的分类到并发更新是否阻塞(最新推荐)

 更新时间:2025年12月31日 09:26:46   作者:蜂蜜黄油呀土豆  
本文介绍了MySQL的锁机制,包括全局锁、表锁和行锁,并详细分析了这些锁在并发场景下的表现和影响,感兴趣的朋友跟随小编一起看看吧

引言

在并发场景下,数据库需要解决三类核心问题:

  • 数据一致性:不能读到脏数据
  • 并发安全:多个事务同时修改数据时不出错
  • 性能权衡:在一致性和并发性能之间取得平衡

MySQL(准确来说是 InnoDB 存储引擎)通过一整套多层次锁机制来解决这些问题。这些锁并不是“随便加的”,而是紧密围绕 事务隔离级别、索引结构、并发访问模式 设计出来的。

本文将从三个层次展开:

  • MySQL 有哪些锁?
  • 表锁和行锁到底解决了什么问题?
  • 真实并发场景下,SQL 会不会阻塞?为什么?

一、MySQL 有哪些锁?——三大类锁体系

作用范围来看,MySQL 的锁可以分为三大类:

全局锁 → 表级锁 → 行级锁

全局锁(Global Lock)

1.1 什么是全局锁?

全局锁会锁住整个数据库实例,使其进入只读状态。

最典型的加锁方式是:

FLUSH TABLES WITH READ LOCK;

此时:

  • 所有表都只能读
  • 所有写操作(INSERT / UPDATE / DELETE)都会被阻塞
1.2 全局锁的典型使用场景

逻辑备份

mysqldump --single-transaction ...

在早期 MySQL 版本中,为了保证备份过程中数据一致,通常需要加全局读锁。

1.3 全局锁的问题
  • 整库不可写
  • 在线业务几乎不可接受

现在更多使用 MVCC + 一致性快照,避免全局锁

表级锁(Table-level Lock)

表级锁的作用范围是 单张表,开销小,但并发能力有限。

2.1 表锁(Table Lock)

LOCK TABLE t_order READ;
LOCK TABLE t_order WRITE;
  • READ:其他线程可读,不可写
  • WRITE:其他线程既不可读,也不可写

InnoDB 一般不推荐使用显式表锁

2.2 元数据锁(MDL,Metadata Lock)

MDL 是自动加的锁,很多人“被它坑过,但不知道是它”。

  • 对表进行 CRUD → 加 MDL 读锁
  • 对表进行 DDL(ALTER / DROP) → 加 MDL 写锁

读写互斥!

-- 事务 A
SELECT * FROM t_order; -- 持有 MDL 读锁
-- 事务 B
ALTER TABLE t_order ADD COLUMN xxx; -- 等待

线上 DDL 阻塞业务的元凶

2.3 意向锁(Intention Lock)

意向锁是 InnoDB 行锁的“前置声明”,存在于表级别。

  • 意向共享锁(IS)
  • 意向排他锁(IX)

作用:让表锁与行锁能快速判断是否冲突

“我要锁某几行了,你别直接给我整个表上锁”

行级锁(Row-level Lock)

行锁是 InnoDB 的核心竞争力,并发性能的关键。

3.1 记录锁(Record Lock)

锁住 某一条索引记录

  • S 锁(共享锁):读
  • X 锁(排他锁):写
SELECT * FROM t_order WHERE id = 10 LOCK IN SHARE MODE;
UPDATE t_order SET status = 1 WHERE id = 10;
3.2 间隙锁(Gap Lock)

锁住的是 索引区间之间的“空隙”

  • 只存在于 RR(可重复读)隔离级别
  • 目的:防止幻读
SELECT * FROM t_order WHERE id BETWEEN 10 AND 20 FOR UPDATE;

锁住 (10, 20) 这个区间,阻止插入新记录

间隙锁之间是兼容的

3.3 临键锁(Next-Key Lock)

Next-Key Lock = Record Lock + Gap Lock

  • 锁住记录本身
  • 同时锁住前面的间隙

这是 InnoDB RR 隔离级别的默认加锁策略

二、表锁和行锁解决了什么问题?

锁类型解决的问题
表锁DDL 与 DML 冲突
行锁并发更新同一行
间隙锁防止幻读
临键锁范围查询 + 插入一致性

本质目标只有一个:
在并发环境下,保证事务隔离语义成立

三、并发场景实战分析:到底会不会阻塞?

场景 1:两个事务同时 UPDATE 同一条记录

UPDATE t_order SET status = 1 WHERE id = 100;

会阻塞

  • 第一个事务获取 id=100 的 X 锁
  • 第二个事务只能等待

场景 2:更新不同主键的记录

-- 事务 A
UPDATE t_order SET status = 1 WHERE id = 100;
-- 事务 B
UPDATE t_order SET status = 1 WHERE id = 200;

不会阻塞

  • 行锁粒度是“记录级”
  • 主键索引精确定位

场景 3:更新主键范围(存在索引)

UPDATE t_order SET status = 1 WHERE id BETWEEN 100 AND 200;
  • 加的是 Next-Key Lock
  • 会锁住区间

其他事务在该区间 INSERT 会被阻塞

场景 4:WHERE 条件未命中索引

UPDATE t_order SET status = 1 WHERE order_no = 'xxx';

如果 order_no 没有索引

  • InnoDB 会 全表扫描
  • 锁升级为 锁住大量记录甚至整表

这才是线上“莫名其妙阻塞”的根源

总结

锁不是 SQL 层决定的,而是:

SQL + 索引 + 隔离级别 + 引擎共同作用的结果

关键记忆点

  • RR 隔离级别 ≠ 只有行锁
  • Next-Key Lock 是默认策略
  • 无索引 ≈ 放弃行锁优势
  • MDL 是 DDL 阻塞的幕后黑手

实战建议

  • 所有更新 SQL 必须走索引
  • 范围更新要评估间隙锁影响
  • 线上 DDL 使用 Online DDL
  • 用 show engine innodb status 排查锁等待

到此这篇关于MySQL 锁机制全解析:从锁的分类到并发更新是否阻塞的文章就介绍到这了,更多相关mysql 锁机制内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • Mysql子查询IN中使用LIMIT应用示例

    Mysql子查询IN中使用LIMIT应用示例

    有3张表,infor信息表,mconfig物料配置表,maaply物料申请表,要求是读出申请表中哪些人申请哪些物料,接下来为大家介绍下Mysql子查询IN中使用LIMIT
    2014-01-01
  • MySQL数据库实验实现简单数据库应用系统设计

    MySQL数据库实验实现简单数据库应用系统设计

    这篇文章主要介绍了MySQL数据库实验实现简单数据库应用系统设计,文章通过理解并能运用数据库设计的常见步骤来设计满足给定需求的概念模和关系数据模型展开详情,需要的朋友可以参考一下
    2022-06-06
  • MySQL Proxy的安装及基本命令使用教程

    MySQL Proxy的安装及基本命令使用教程

    这篇文章主要介绍了MySQL Proxy的安装及基本命令使用教程,MySQL Proxy通常被用作实现读写分离,需要的朋友可以参考下
    2015-12-12
  • mysql5.6.8源码安装过程

    mysql5.6.8源码安装过程

    这篇文章主要介绍了mysql5.6.8源码安装过程,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2018-01-01
  • MAC下Mysql5.7.10版本修改root密码的方法

    MAC下Mysql5.7.10版本修改root密码的方法

    这篇文章主要介绍了MAC下Mysql5.7.10版本修改root密码的方法,非常不错,具有参考借鉴价值,需要的朋友可以参考下
    2017-03-03
  • MySQL创建用户的三种方法

    MySQL创建用户的三种方法

    在对MySQL的日常管理和操作中,为了避免有人恶意使用root用户控制数据库,尽可能地不用或少用 root 用户登录系统,本文主要介绍了MySQL创建用户的三种方法,感兴趣的可以了解一下
    2023-08-08
  • 关于MySQL中的 like操作符详情

    关于MySQL中的 like操作符详情

    这篇文章主要介绍了MySQL之like操作符,当对未知或者说知道某一部分的值进行过滤时,可以使用like操作符;like操作符用于模糊匹配。下面我们一起进入文章看看文章是我详细内容
    2021-11-11
  • MySQL 迁移OB Oracle场景中自增主键实践操作

    MySQL 迁移OB Oracle场景中自增主键实践操作

    这篇文章主要介绍了MySQL 迁移OB Oracle场景中自增主键实践操作详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-10-10
  • 解决Navicat导入数据库数据结构sql报错datetime(0)的问题

    解决Navicat导入数据库数据结构sql报错datetime(0)的问题

    这篇文章主要介绍了解决Navicat导入数据库数据结构sql报错datetime(0)的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-07-07
  • 怎样设置才能允许外网访问MySQL

    怎样设置才能允许外网访问MySQL

    大多数情况下,mysql数据库只要本机访问就可以了,这样的话,默认安装就OK,但是如果需要外网访问mysql数据库的话,应该如何操作呢,想知道的话,就好好看看下面的介绍吧
    2014-08-08

最新评论