当前位置: 首页 > ai >正文

数据库学习(三)——MySQL锁

一、MySQL锁

当多个用户或进程并发操作数据库时,为了避免数据冲突、脏读、不可重复读、幻读等问题,数据库通过锁机制来保证数据的一致性完整性

MySQL 锁的总体分类:

分类维度类型说明
按作用对象表级锁(Table Lock)锁住整张表
行级锁(Row Lock)锁住一行数据
按锁粒度全局锁、数据库锁、表锁、行锁粒度逐级细化
按加锁方式共享锁(S锁)允许读,不允许写
排他锁(X锁)允许写,其他读写都被阻塞
按使用方式显式锁、隐式锁手动加锁 or 自动加锁
按存储引擎实现不同引擎的锁实现机制不同InnoDB vs MyISAM

二、MySQL 中的锁类型

1. 表级锁(Table Lock)

  • 作用于整张表
  • 典型场景:MyISAM 存储引擎、ALTER TABLELOCK TABLES
  • 效率高、开销小,但并发性差。

示例:

-- 给表 user 加上 写锁(WRITE LOCK)。
LOCK TABLES user WRITE;
-- 操作
UNLOCK TABLES;

2. 行级锁(Row Lock)

  • 作用于某一行数据
  • InnoDB 引擎支持,适合高并发读写场景;
  • 粒度小,并发性好,但管理成本高。

示例——隐式加锁:

BEGIN;
SELECT * FROM orders WHERE id = 10 FOR UPDATE; -- X锁
COMMIT;

3. InnoDB 的事务级锁(最常用)

(1)共享锁(S锁, Shared Lock)

  • 多个事务可以并发加 S 锁,读取数据;
  • 但不能加 X 锁,写操作会被阻塞。

示例:

SELECT * FROM products WHERE id=1 LOCK IN SHARE MODE;

注意:

  • 为查询到的行加上共享锁(S锁),直到事务结束(提交或回滚)。

(2)排他锁(X锁, Exclusive Lock)

  • 独占锁:读写都阻塞其他事务;
  • 常用于修改数据。

示例:

SELECT * FROM products WHERE id=1 FOR UPDATE;

注意:

  • 查询的同时,对满足条件的记录加排他锁,防止其他事务对该记录进行修改或加锁读取。

(3)意向锁(Intention Lock)

  • InnoDB 自动加的表级锁,用于指示事务是否准备加行锁;
  • 无需手动操作;
  • 分为:意向共享锁(IS)意向排他锁(IX)

4. 其他特殊类型

(1)Gap Lock(间隙锁)

  • 锁住的是数据之间的“间隙”防止幻读
  • 例如:SELECT * FROM t WHERE age > 30 FOR UPDATE;

(2)Next-Key Lock(记录锁 + 间隙锁)

  • InnoDB 默认的锁,避免幻读;
  • 锁住“记录本身 + 相邻间隙”。

(3)自增长锁(Auto-Inc Lock)

  • 针对 AUTO_INCREMENT 字段的内部锁
  • 防止并发插入冲突。

三、锁的工作原理(以 InnoDB 为例)

  • 当事务执行写操作时,InnoDB 会在相关行上加排他锁(X锁)
  • 读操作默认不加锁(基于 MVCC 实现快照读),除非使用 SELECT ... FOR UPDATELOCK IN SHARE MODE 强制加锁。
  • 当一个事务对某行加了 X 锁,其他事务必须等待锁释放后才能访问该行。

四、常见锁相关语句

  • SELECT ... FOR UPDATE:对查询到的行加排他锁。
  • SELECT ... LOCK IN SHARE MODE:对查询到的行加共享锁。
  • LOCK TABLESUNLOCK TABLES:对表加锁和解锁(表锁)。

五、常见问题

1. 死锁(Deadlock)

  • 多个事务互相等待对方持有的锁,导致程序无限等待。

  • InnoDB 能自动检测死锁,并回滚其中一个事务。

  • 解决策略:

    • 保持事务尽量短小
    • 尽量避免长时间锁住资源
    • 按照固定顺序访问表和行。

2. 脏写(Dirty Write)

定义:
一个事务写入了另一个未提交事务修改的数据,这会导致数据污染或逻辑错误

示例:

-- 事务A:
BEGIN;
UPDATE users SET name = 'Alice' WHERE id = 1;-- 事务B 在事务A未提交时:
BEGIN;
UPDATE users SET name = 'Bob' WHERE id = 1;

注意:

  • 如果数据库允许这样操作,就是脏写。
  • 假如事务A回滚了,但事务B的修改已提交,数据将不一致。

解决:

  • MySQL InnoDB 的默认隔离级别 REPEATABLE READ 已避免脏写。
  • 即使设置为 READ COMMITTED,也不允许脏写。

3. 幻读(Phantom Read)

定义:
在同一个事务中,前后两次相同条件的查询,返回的记录数不一样,即第二次出现了“幻影”数据。

示例:

-- 事务A:
BEGIN;
SELECT * FROM orders WHERE amount > 1000;-- 事务B 插入新数据:
INSERT INTO orders(amount) VALUES(2000); -- 满足条件-- 事务A 再次执行:
SELECT * FROM orders WHERE amount > 1000;
-- 返回了比上次多一行

注意:

  • READ COMMITTEDREPEATABLE READ 级别下会出现幻读。

六、锁的隔离级别

隔离级别说明是否使用行锁
READ UNCOMMITTED允许脏读无锁或极少
READ COMMITTED不允许脏读读操作不加锁(快照读)
REPEATABLE READ可重读InnoDB 默认隔离级别读操作快照读,写操作加行锁
SERIALIZABLE串行化,最高隔离级别,所有读操作加锁读写都加锁
http://www.xdnf.cn/news/12836.html

相关文章:

  • for循环应用
  • 【西门子杯工业嵌入式-6-ADC采样基础】
  • 详细叙述一下Spring如何创建bean
  • Python训练营打卡DAY48
  • 华为IP(8)(OSPF开放最短路径优先)
  • 树状数组学习笔记
  • 振动力学:无阻尼多自由度系统(受迫振动)
  • SQL进阶之旅 Day 21:临时表与内存表应用
  • Spring MVC请求处理流程和DispatcherServlet机制解析
  • 【Go语言基础【18】】Map基础
  • 2025-04-28-堆、栈及其应用分析
  • 算法专题七:分治
  • 【CATIA的二次开发23】抽象对象Document涉及文档激活控制的方法
  • serv00 ssh登录保活脚本-邮件通知版
  • 【构建】CMake 常用函数和命令清单
  • leetcode189-轮转数组
  • Prefix Caching 详解:实现 KV Cache 的跨请求高效复用
  • c++对halcon的动态链接库dll封装及调用(细细讲)
  • 【CSS-8】深入理解CSS选择器权重:掌握样式优先级的关键
  • 【拆机系列】暴力拆解AOC E2270SWN6液晶显示屏
  • Python训练营打卡Day48(2025.6.8)
  • 【LangChain4J】LangChain4J 第三弹:多模态与文生图的实现
  • leetcode_56 合并区间
  • el-table的select回显问题
  • 图解JavaScript原型:原型链及其分析 | JavaScript图解
  • Alight Motion汉化版:视频剪辑,轻松上手
  • odoo17 反常下表引用上表并能修改
  • 在数字工厂实施过程中,如何学会通过梳理流程的思想来分析解决问题
  • 2014-2023年 最新绿色债券数据
  • 大数据CSV导入MySQL