解读Mysql中的共享锁,排它锁与意向锁

共享锁与排它锁

本文执行的事务都在repeatable read隔离级别(mysql8 下的默认级别)下执行的

假设有一个 accounts 表:

diff 复制代码
+----+---------+
| id | balance |
+----+---------+
|  1 | 1000.00 |
|  2 |  500.00 |
+----+---------+

共享锁

测试 1: 被锁行在事务提交前只能被读取, 其他进程无法修改被锁行数据

Shell A Shell B
start transaction;
select * from accounts where id=1 for share;
start transaction;
select * from accounts where id=1;
正常出结果
update accounts set balance=balance+100 where id=1;
暂停执行, 等待 Shell A 提交
commit; (自动执行 update 语句)

测试 2: 被锁行在事务提交前, 其他进程可正常添加共享锁, 但无法修改行数据

Shell A Shell B
start transaction;
select * from accounts where id=1 for share;
start transaction;
select * from accounts where id=1 for share;
正常出结果,并获取共享锁
update accounts set balance=balance+100 where id=1;
暂停执行, 等待 Shell A 提交
commit; (自动执行 update 语句)

测试 3: 被锁行在事务提交前, 其他进程无法添加排他锁

Shell A Shell B
start transaction;
select * from accounts where id=1 for share;
start transaction;
select * from accounts where id=1 for update;
暂停执行, 等待 Shell A 提交
commit; (自动执行 select 语句)

测试 4: 被锁行在事务内执行了非 select 语句, 在事务提交前, 其他进程无法添加共享锁或排它锁.(排它锁这里没有案例, 但共享锁都加不了, 排它锁肯定也加不了)

Shell A Shell B
start transaction;
update accounts set balance=balance+100 where id=1;
start transaction;
select * from accounts where id=1 for share;
暂停执行, 等待 Shell A 提交
commit; (自动执行 select 语句)

排它锁

测试 1: 被锁行在事务提交前, 其他进程只能读取, 无法修改被锁行数据

Shell A Shell B
start transaction;
select * from accounts where id=1 for update;
select * from accounts where id=1;
正常出结果
update accounts set balance=balance+100 where id=1;
暂停执行, 等待 Shell A 提交
commit; (自动执行 update 语句)

测试 2: 被锁行在事务提交前, 其他进程不能添加共享锁

Shell A Shell B
start transaction;
select * from accounts where id=1 for update;
start transaction;
select * from accounts where id=1 for share;
暂停执行, 等待 Shell A 提交
commit; (自动执行 select 语句)

测试 3: 被锁行在事务提交前, 其他进程不能添加排他锁

Shell A Shell B
start transaction;
select * from accounts where id=1 for update;
start transaction;
select * from accounts where id=1 for update;
暂停执行, 等待 Shell A 提交
commit; (自动执行 select 语句)

意向锁

意向锁是一种用于协调并发访问的锁定机制,它不直接锁定数据,而是用于指示事务的意图,即该事务是否打算锁定数据。

考虑一个数据库中的多个表或数据行,多个事务可能需要对这些数据进行读取或修改。意向锁的作用是协助数据库系统了解事务的意向,以便更有效地管理锁定。

举个例子:

事务 A 打算对某一行数据进行写入操作,因此它会请求排他锁(Exclusive Lock)。 同时,事务 B 打算对同一行数据进行读取操作,因此它会请求共享锁(Shared Lock)。 这里的问题是,数据库系统如何知道有其他事务正在打算锁定这一行数据呢?这时就引入了意向锁:

事务 A 在请求排他锁之前会请求意向排他锁(Intent Exclusive Lock)来指示它的意向是锁定数据。 事务 B 在请求共享锁之前会请求意向共享锁(Intent Shared Lock)来指示它的意向是锁定数据。 意向锁并不直接影响数据,它只是一种标记,告诉数据库系统有哪些事务打算锁定数据。这有助于系统更好地管理并发操作,避免冲突,提高性能。

所以,意向锁是一种数据库内部的机制,用于协助事务之间的锁定协调,以确保数据的一致性和并发性。它们不是直接应用到数据上的锁定。

意向锁(Intent Lock)通常是数据库系统内部管理的,而不需要显式地添加或控制。平时主要与共享锁和排他锁打交道,而数据库系统会自动处理意向锁以协调事务之间的并发访问。

当一个事务请求共享锁或排他锁时,数据库系统会自动添加相应的意向锁。例如,当事务请求排他锁时,数据库系统会添加意向排他锁(IX 锁)以指示它的意向是锁定数据。当事务请求共享锁时,系统会添加意向共享锁(IS 锁)。

开发中只需关注获取所需的锁(共享锁或排他锁)以及正确地释放这些锁,数据库系统会负责管理和处理意向锁。

相关推荐
斗-匕4 分钟前
MySQL 三大日志详解
数据库·mysql·oracle
代码中の快捷键10 分钟前
MySQL数据库存储引擎
数据库·mysql
六月闻君1 小时前
MySQL 报错:1137 - Can‘t reopen table
数据库·mysql
白云如幻1 小时前
SQL99版链接查询语法
数据库·sql·mysql
爱吃烤鸡翅的酸菜鱼2 小时前
MySQL初学之旅(4)表的设计
数据库·sql·mysql·database
计算机毕设指导62 小时前
基于 SpringBoot 的作业管理系统【附源码】
java·vue.js·spring boot·后端·mysql·spring·intellij-idea
打鱼又晒网4 小时前
【MySQL】数据库精细化讲解:内置函数知识穿透与深度学习解析
数据库·mysql
tatasix5 小时前
MySQL UPDATE语句执行链路解析
数据库·mysql
天海华兮5 小时前
mysql 去重 补全 取出重复 变量 函数 和存储过程
数据库·mysql
武子康7 小时前
大数据-231 离线数仓 - DWS 层、ADS 层的创建 Hive 执行脚本
java·大数据·数据仓库·hive·hadoop·mysql