最近、ある同僚が本番環境で MySQL のデッドロックの問題に遭遇したため、問題の解決を手伝った後、特別に 1 週間かけてすべての MySQL ロックを整理しました. 今日は、MySQL ロックについて一緒に話しましょう.
ステートメント: この記事は、MySQL 8.0.30 バージョン、InnoDB エンジンに基づいています。
MySQL データベース ロック設計の本来の意図は、同時実行性の問題に対処し、データ セキュリティを確保することです。MySQL データベースのロックは、次の 3 つの次元から分割できます。
- ロックの使用に応じて、MySQL ロックは共有ロックと排他ロックに分けることができます。
- ロックの範囲に応じて、MySQL ロックは大まかに 3 つのカテゴリに分類できます。グローバル ロック、テーブル レベル ロック、および行ロックです。
- イデオロギーの観点から、MySQL ロックは悲観的ロックと楽観的ロックの 2 つのタイプに分けることができます。
最初に共有ロックと排他ロックについて説明し、次にグローバル ロック、テーブル レベル ロック、および行ロックについて説明します (これら 3 つのロックには共有ロックと排他ロックがあるため)。楽観的ロック。
共有ロックと排他ロック
1.共有ロック
共有ロック、共有ロック、読み取りロックとも呼ばれます。つまり、オブジェクトがロックされている場合、他のトランザクションはオブジェクトを読み取ることができ、他のトランザクションはオブジェクトから共有ロックを再度取得することができますが、オブジェクトに書き込むことはできません。
ロック方法は次のとおりです。
# 方式1
select ... lock in share mode;
# 方式2
select ... for share;
トランザクション T1 がオブジェクトの共有 (S) ロックを保持しており、トランザクション T2 がオブジェクトのロックを再度取得する必要がある場合、次の 2 つの状況が発生します。
- T2 がオブジェクトの共有 (S) ロックを取得すると、すぐにロックを取得できます。
- T2 がオブジェクトの排他 (X) ロックを取得した場合、ロックを取得できません。
上記の 2 つの状況をよりよく理解するために、次の実行シーケンス フローと図の例を参照できます。
ユーザー テーブルに共有ロックを追加する
ロックスレッド sessionA | スレッド B セッション B |
---|---|
#オープン トランザクション の開始; |
|
# user テーブル全体に共有ロックを追加 select * from user lock in share mode; |
|
#user テーブルの共有ロックを取得します。選択操作が正常に実行されました select * from user; |
|
#ユーザー テーブルの排他ロックを取得できませんでした 。id = 1 のユーザーからの削除操作がブロックされました。 |
|
#Commit transaction# ユーザーテーブルの共有ロックが解除されます 。 |
|
# user テーブルの排他ロックの取得に成功し、削除操作で ok delete from user where id = 1;が実行されます。 |
user テーブルの id=3 の行に共有ロックを追加します
ロックスレッド sessionA | スレッド B セッション B | スレッド C セッション C |
---|---|---|
#オープン トランザクション の開始; |
||
# user テーブルの id=3 の行に共有ロックを追加 select * from user where id = 3 lock in share mode; |
||
#user テーブルの共有ロックを取得 id=3 行 ok #select 操作が正常に実行されました select * from user where id=3; |
#user テーブルの共有ロックを取得 id=3 行 ok #select 操作が正常に実行されました select * from user where id=3; |
|
# ユーザー テーブルの排他ロックを取得します id=3 行に失敗しました #delete 操作がブロックされました delete from user where id = 3; |
#user テーブル id=4 行の排他ロックの取得に成功 #delete 操作が正常に実行されました delete from user where id = 4; |
|
#Submit transaction# user テーブル id=3 の行の共有ロックが解除されます commit; |
||
# ユーザー テーブル id=3 行の排他ロックを正常に取得 # ブロックされた削除操作により、 id = 3 のユーザーから ok delete が実行されます。 |
[外部リンクの画像転送に失敗しました。ソース サイトには盗難防止リンク メカニズムがある可能性があります。画像を保存して直接アップロードすることをお勧めします (img-Zg3nYfxW-1665050356867)(https://yuanjava.cn/assets/md/ mysql/share-lock-row .png)]
上記の 2 つの例から、次のことがわかります。
- 共有ロックがユーザー テーブルに追加されると、他のトランザクションはユーザー テーブルの共有ロックを再度取得できますが、他のトランザクションはユーザー テーブルの排他ロックを再度取得できず、操作がブロックされます。
- user テーブルの id=3 の行に共有ロックを追加すると、他のトランザクションは user テーブルの id=3 の行の共有ロックを再度取得でき、他のトランザクションはその行の排他ロックの取得に失敗します。ユーザー テーブルに id=3 が再度あり、操作はブロックされますが、トランザクションはユーザー テーブル id!=3 行の排他ロックを再度取得できます。
2.排他ロック
排他ロック、排他ロックは、書き込みロックまたは排他ロックとも呼ばれ、主に、他のトランザクションが現在のロック トランザクションと同じオブジェクトをロックするのを防ぎます。同じオブジェクトには、主に 2 つの意味があります。
- テーブルに排他ロックが追加されると、他のトランザクションはテーブルに対して挿入、更新、削除、変更、削除などの更新操作を実行できなくなります。
- テーブル行に排他ロックが追加されると、他のトランザクションはその行に対して挿入、更新、削除、変更、ドロップなどの更新操作を実行できなくなります。
排他ロックのロック方法は次のとおりです。
select ... for update;
排他ロックをよりよく説明するために、次の実行シーケンス フローとインスタンス図を参照できます。
ユーザー テーブル オブジェクトに排他ロックを追加する
ロックスレッド sessionA | スレッド B セッション B |
---|---|
#オープン トランザクションの開始; | |
#user select * from user for update;のテーブル全体に排他ロックを追加します。 |
|
#user テーブルの共有ロックを取得します。select は正常に実行されました select * from user; |
|
#ユーザー テーブルの排他ロックを取得できませんでした。id=3 のユーザーからの削除操作がブロックされました。 |
|
#コミットトランザクション# ユーザーテーブル commit の排他的解放。 |
|
#获取user表上的排他锁成功,操作执行ok delete from user where id = 1; |
给user表id=3的行对象加排他锁
加锁线程 sessionA | 线程B sessionB | 线程C sessionC |
---|---|---|
#开启事务 begin; |
||
#给user表id=3的行加排他锁 select * from user where id = 3 for update; |
||
#获取user表id=3行上的共享锁ok select * from user where id=3; |
#获取user表id=3行上的共享锁ok select * from user where id=3; |
|
#获取user表id=3行上的排它锁失败 delete from user where id = 3; |
#获取user表id=4行上的排它锁成功 delete from user where id = 4; |
|
#提交事务 #user表id=3的行上排他锁被释放 commit; |
||
#获取user表id=3行上的排它锁成功 #被堵塞的delete操作执行ok delete from user where id = 3; |
全局锁&表级锁&行锁
1. 全局锁
1.1 定义
全局锁,顾名思义,就是对整个数据库实例加锁。它是粒度最大的锁。
1.2 加锁
在MySQL中,通过执行 flush tables with read lock 指令加全局锁:
flush tables with read lock
指令执行完,整个数据库就处于只读状态了,其他线程执行以下操作,都会被阻塞:
- 数据更新语句被阻塞,包括 insert, update, delete语句;
- 数据定义语句被阻塞,包括建表 create table,alter table、drop table 语句;
- 更新操作事务commit语句被阻塞;
1.3 释放锁
MySQl释放锁有2种方式:
- 执行 unlock tables 指令
unlock tables
- 加锁的会话断开,全局锁也会被自动释放
为了更好的说明全局锁,可以参照下面的执行顺序流和实例图:
加锁线程 sessionA | 线程B sessionB |
---|---|
flush tables with read lock; 加全局锁 | |
select user表ok | select user表ok |
insert user表堵塞 | insert user表堵塞 |
delete user表堵塞 | delete user表堵塞 |
drop user 表堵塞 | drop user 表堵塞 |
alter user表 堵塞 | alter user表 堵塞 |
unlock tables; 解锁 | |
被堵塞的修改操作执行ok | 被堵塞的修改操作执行ok |
通过上述的实例可以看出,当加全局锁时,库下面所有的表都处于只能状态,不管是当前事务还是其他事务,对于库下面所有的表只能读,不能执行insert,update,delete,alter,drop等更新操作。
1.4 使用场景
全局锁的典型使用场景是做全库逻辑备份,在备份过程中整个库完全处于只读状态。如下图:
- 假如在主库上备份,备份期间,业务服务器不能对数据库执行更新操作,因此涉及到更新操作的业务就瘫痪了;
- 假如在从库上备份,备份期间,从库不能执行主库同步过来的 binlog,会导致主从延迟越来越大,如果做了读写分离,那么从库上获取数据就会出现延时,影响业务;
从上述分析可以看出,使用全局锁进行数据备份,不管是在主库还是在从库上进行备份操作,对业务总是不太友好。那不加锁行不行?我们可以通过下面还钱转账的例子,看看不加锁会不会出现问题:
- 备份前:账户A 有1000,账户B 有500
- 此时,发起逻辑备份
- 假如数据备份时不加锁,此时,客户端A 发起一个还钱转账的操作:账户A 往账户B 转200
- 当账户A 转出200完成,账户B 转入200 还未完成时,整个数据备份完成
- 如果用该备份数据做恢复,会发现账户A 转出了200,账户B 却没有对应的转入记录,这样就会产生纠纷:A 说我账户少了 200, B 说我没有收到,最后,A,B谁都不干。
既然不加锁会产生错误,加全局锁又会影响业务,那么有没有两全其美的方式呢?
有,MySQL官方自带的逻辑备份工具 mysqldump,具体指令如下:
mysqldump –single-transaction
执行该指令,在备份数据之前会先启动一个事务,来确保拿到一致性视图, 加上 MVCC 的支持,保证备份过程中数据是可以正常更新。但是,single-transaction方法只适用于库中所有表都使用了事务引擎,如果有表使用了不支持事务的引擎,备份就只能用 FTWRL 方法。
2. 表级锁
MySQL 表级锁有两种:
- 表锁
- 元数据锁(metadata lock,MDL)
2.1 表锁
表锁就是对整张表加锁,包含读锁和写锁,由MySQL Server实现,表锁需要显示加锁或释放锁,具体指令如下:
# 给表加写锁
lock tables tablename write;
# 给表加读锁
lock tables tablename read;
# 释放锁
unlock tables;
读锁:代表当前表为只读状态,读锁是一种共享锁。需要注意的是,读锁除了会限制其它线程的操作外,也会限制加锁线程的行为,具体限制如下:
- 加锁线程只能对当前表进行读操作,不能对当前表进行更新操作,不能对其它表进行所有操作;
- 其它线程只能对当前表进行读操作,不能对当前表进行更新操作,可以对其它表进行所有操作;
为了更好的说明读锁,可以参照下面的执行顺序流和实例图:
加锁线程 sessionA | 线程B sessionB |
---|---|
#给user表加读锁 lock tables user read; |
|
select user表 ok | select user表 ok |
insert user表被拒绝 | insert user表堵塞 |
insert address表被拒绝 | insert address表ok |
select address表被拒绝 | alter user表堵塞 |
unlock tables; 释放锁 | |
被堵塞的修改操作执行ok |
写锁:写锁是一种独占锁,需要注意的是,写锁除了会限制其它线程的操作外,也会限制加锁线程的行为,具体限制如下:
- 加锁线程对当前表能进行所有操作,不能对其它表进行任何操作;
- 其它线程不能对当前表进行任何操作,可以对其它表进行任何操作;
为了更好的说明写锁,可以参照下面的执行顺序流和实例图:
加锁线程 sessionA | 线程B sessionB |
---|---|
#给user表加写锁 lock tables user write; |
|
select user表 ok | select user表 ok |
insert user表被拒绝 | insert user表堵塞 |
insert address表被拒绝 | insert address表ok |
select address表被拒绝 | alter user表堵塞 |
unlock tables; 释放锁 | |
堵塞在user表的上更新操作执行ok |
2.2 MDL元数据锁
元数据锁:metadata lock,简称MDL,它是在MySQL 5.5版本引进的。元数据锁不用像表锁那样显式的加锁和释放锁,而是在访问表时被自动加上,以保证读写的正确性。加锁和释放锁规则如下:
- MDL读锁之间不互斥,也就是说,允许多个线程同时对加了 MDL读锁的表进行CRUD(增删改查)操作;
- MDL写锁,它和读锁、写锁都是互斥的,目的是用来保证变更表结构操作的安全性。也就是说,当对表结构进行变更时,会被默认加 MDL写锁,因此,如果有两个线程要同时给一个表加字段,其中一个要等另一个执行完才能开始执行。
- MDL读写锁是在事务commit之后才会被释放;
为了更好的说明 MDL读锁规则,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 其它线程 sessionB |
---|---|
开启事务 begin; |
|
select user表,user表会默认加上MDL读锁 | |
select user表ok | select user表ok |
insert user表ok | insert user表ok |
update user表ok | update user表ok |
delete user表ok | delete user表ok |
alter user表,获取MDL写锁失败,操作被堵塞 | |
commit;提交事务,MDL读锁被释放 | |
被堵塞的修改操作执行ok |
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-sIyVqMzT-1665050356868)(https://yuanjava.cn/assets/md/mysql/MDL-read-lock.png)]
为了更好的说明 MDL写锁规则,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 线程B sessionB | 线程C sessionC |
---|---|---|
#开启事务 begin; |
||
#user表会默认加上MDL读锁 select user表, |
||
select user表ok | select user表ok | select user表ok |
#获取MDL写锁失败 alter user表操作被堵塞 |
#获取MDL读锁失败 select * from user; |
|
提交事务,MDL读锁被释放 | ||
#MDL写锁被释放 被堵塞的alter user操作执行ok |
||
#被堵塞的select 操作执行ok |
2.3 意向锁
由于InnoDB引擎支持多粒度锁定,允许行锁和表锁共存,为了快速的判断表中是否存在行锁,InnoDB推出了意向锁。
意向锁,Intention lock,它是一种表锁,用来标识事务打算在表中的行上获取什么类型的锁。
不同的事务可以在同一张表上获取不同种类的意向锁,但是第一个获取表上意向排他(IX) 锁的事务会阻止其它事务获取该表上的任何 S锁 或 X 锁。反之,第一个获得表上意向共享锁(IS) 的事务可防止其它事务获取该表上的任何 X 锁。
意向锁通常有两种类型:
- 意向共享锁(IS),表示事务打算在表中的各个行上设置共享锁。
- 意向排他锁(IX),表示事务打算对表中的各个行设置排他锁。
意向锁是InnoDB自动加上的,加锁时遵从下面两个协议:
- 事务在获取表中行的共享锁之前,必须先获取表上的IS锁或更强的锁。
- 事务在获取表中行的排他锁之前,必须先获取表上的IX锁。
为了更好的说明意向共享锁,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 线程B sessionB |
---|---|
#开启事务 begin; |
|
#user表id=6加共享行锁 ,默认user表会 加上IS锁 select * from user where id = 6 for share; |
|
# 观察IS锁 select * from performance_schema.data_locks\G |
加锁线程 sessionA | 线程B sessionB |
---|---|
#开启事务 begin; |
|
#user表id=6加排他锁,默认user表会 加上IX锁 select * from user where id = 6 for update; |
|
# 观察IX锁 select * from performance_schema.data_locks\G |
2.4 AUTO-INC锁
AUTO-INC锁是一种特殊的表级锁,当表中有AUTO_INCREMENT的列时,如果向这张表插入数据时,InnoDB会先获取这张表的AUTO-INC锁,等插入语句执行完成后,AUTO-INC锁会被释放。
AUTO-INC锁可以使用innodb_autoinc_lock_mode变量来配置自增锁的算法,innodb_autoinc_lock_mode变量可以选择三种值如下表:
innodb_autoinc_lock_mode | 含义 |
---|---|
0 | 传统锁模式,采用 AUTO-INC 锁 |
1 | 连续锁模式,采用轻量级锁 |
2 | 交错锁模式(MySQL8默认),AUTO-INC和轻量级锁之间灵活切换 |
为了更好的说明意AUTO-INC锁,可以参照下面的顺序执行流和实例图:
2.5 锁的兼容性
下面的图表总结了表级锁类型的兼容性
X | IX | S | IS | |
---|---|---|---|---|
X | 冲突 | 冲突 | 冲突 | 冲突 |
IX | 冲突 | 兼容 | 冲突 | 兼容 |
S | 冲突 | 冲突 | 兼容 | 兼容 |
IS | 冲突 | 兼容 | 兼容 | 兼容 |
3. 行锁
行锁是针对数据表中行记录的锁。MySQL 的行锁是在引擎层实现的,并不是所有的引擎都支持行锁,比如,InnoDB引擎支持行锁而 MyISAM引擎不支持。
InnoDB 引擎的行锁主要有三类:
- Record Lock: 记录锁,是在索引记录上加锁;
- Gap Lock:间隙锁,锁定一个范围,但不包含记录;
- Next-key Lock:Gap Lock + Record Lock,锁定一个范围(Gap Lock实现),并且锁定记录本身(Record Lock实现);
3.1 Record Lock
Record Lock:记录锁,是针对索引记录的锁,锁定的总是索引记录。
例如,select id from user where id = 1 for update; for update 就显式在索引id上加行锁(排他锁),防止其它任何事务 update或delete id=1 的行,但是对user表的insert、alter、drop操作还是可以正常执行。
为了更好的说明 Record Lock锁,可以参照下面的执行顺序流和实例图:
加锁线程 sessionA | 线程B sessionB | 线程B sessionC |
---|---|---|
#开启事务 begin; |
||
给user表id=1加写锁 select id from user where id = 1 for update; |
||
update user set name = ‘name121’ where id = 1; |
||
查看 InnoDB监视器中记录锁数据 show engine innodb status\G |
||
commit提交事务 record lock 被释放 |
||
被堵塞的update操作执行ok |
3.2 Gap Lock
Gap Lock:间隙锁,锁住两个索引记录之间的间隙上,由InnoDB隐式添加。比如(1,3) 表示锁住记录1和记录3之间的间隙,这样记录2就无法插入,间隙可能跨越单个索引值、多个索引值,甚至是空。
为了更好的说明 Gap Lock间隙锁,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 线程B sessionB | 线程C sessionC |
---|---|---|
#开启事务 begin; |
||
加锁 select * from user where age = 10 for share; |
||
insert into user(id,age) values(2,20); | ||
#查看 InnoDB监视器中记录锁数据 show engine innodb status\G |
||
commit提交事务 Gap Lock被释放 |
||
被堵塞的insert操作执行ok |
上图中,事务A(sessionA)在加共享锁的时候产生了间隙锁(Gap Lock),事务B(sessionB)对间隙中进行insert/update操作,需要先获取排他锁(X),导致阻塞。事务C(sessionC)通过"show engine innodb status\G" 指令可以查看到间隙锁的存在。需要说明的,间隙锁只是锁住间隙内部的范围,在间隙外的insert/update操作不会受影响。
Gap Lock锁,只存在于可重复读隔离级别,目的是为了解决可重复读隔离级别下幻读的现象。
3.3 Next-Key Lock
Next-Key锁,称为临键锁,它是Record Lock + Gap Lock的组合,用来锁定一个范围,并且锁定记录本身锁,它是一种左开右闭的范围,可以用符号表示为:(a,b]。
为了更好的说明 Next-Key Lock间隙锁,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 线程B sessionB | 线程C sessionC | 线程D sessionD |
---|---|---|---|
#开启事务 begin; |
|||
加锁 select * from user where age = 10 for share; |
|||
#获取锁失败,insert操作被堵塞 insert into user(id,age) values(2,20); |
|||
update user set name=‘name1’ where age = 10; |
#查看 InnoDB监视器中记录锁数据 show engine innodb status\G |
||
提交事务Gap Lock被释放 commit |
|||
被堵塞的insert操作执行ok | 被堵塞的update操作执行ok |
上图中,事务A(sessionA)在加共享锁的时候产生了间隙锁(Gap Lock),事务B(sessionB)对间隙中进行insert操作,需要先获取排他锁(X),导致阻塞。
事务C(sessionC)对间隙中进行update操作,需要先获取排他锁(X),导致阻塞。
事务D(sessionD)通过"show engine innodb status\G" 指令可以查看到间隙锁的存在。需要说明的,间隙锁只是锁住间隙内部的范围,在间隙外的insert/update操作不会受影响。
3.4 Insert Intention Lock
插入意向锁,它是一种特殊的间隙锁,特指插入操作产生的间隙锁。
为了更好的说明 Insert Intention Lock锁,可以参照下面的顺序执行流和实例图:
加锁线程 sessionA | 线程B sessionB | 线程C sessionC |
---|---|---|
#开启事务 begin; |
||
加锁 select * from user where age = 10 for share; |
||
#获取锁失败,insert操作被堵塞 insert into user(id,age) values(2,20); |
||
#查看 InnoDB监视器中记录锁数据 show engine innodb status\G |
||
commit提交事务 Gap Lock被释放 |
||
#被堵塞的insert操作执行ok insert into user(id,age) values(2,20); |
乐观锁&悲观锁
MySQL では、悲観的ロックであろうと楽観的ロックであろうと、概念に対する人間の考え方の一種の抽象化であり、それら自体は MySQL が提供するロック メカニズムを使用して実装されます。実際、MySQL データに加えて、Java 言語には楽観的ロックと悲観的ロックの概念もあります。
- 悲観的ロックは次のように理解できます。レコードを変更する前に、まずレコードに排他ロックを追加し (排他ロック)、最初にロックを取得してからデータを操作する戦略を採用します。これにより、デッドロックが発生する可能性があります。
- 悲観的ロックと比較して、楽観的ロックは通常、データベースのロック メカニズムを使用しませんが、バージョン番号の比較などの操作を使用するため、楽観的ロックはデッドロックの問題を引き起こしません。
デッドロックとデッドロック検出
並行システム内の異なるスレッドが周期的なリソースの依存関係を持ち、関連するスレッドがすべて他のスレッドがリソースを解放するのを待っている場合、これらのスレッドはデッドロックと呼ばれる無限の待機状態になります。次のコマンドでデッドロックを確認できます
show engine innodb status\G
デッドロックが発生した場合、次の 2 つの戦略があります。
- 1 つの戦略は、先に進んでタイムアウトになるまで待つことです。このタイムアウトはパラメーター innodb_lock_wait_timeout で設定でき、InnoDB の innodb_lock_wait_timeout のデフォルト値は 50 秒です。
- もう 1 つの戦略は、デッドロックの検出を開始し、デッドロックが見つかった後、デッドロック チェーン内のトランザクションをアクティブにロールバックして、他のトランザクションが引き続き実行できるようにすることです。パラメータ innodb_deadlock_detect を on に設定して、デッドロック検出を有効にします。
要約する
この記事では、MySQL 8.0.30 バージョンと InnoDB エンジンに基づいて MySQL のロックについて説明します.各ロックには、特定の使用シナリオがあります。
MySQL を頻繁に扱う Java プログラマーとして、MySQL ロックの理解が深ければ深いほど、高性能 SQL ステートメントを作成するのに役立ちます。