Mysql select for update 的加锁机制?

2023-09-24 05:39:51 +08:00
 hjcpnr
如下表,age 上有唯一索引,隔离级别 RC

CREATE TABLE `test` (
`id` int NOT NULL AUTO_INCREMENT,
`age` char(10) NOT NULL,
`name` char(10) NOT NULL,
PRIMARY KEY (`id`),
UNIQUE KEY `idx_age` (`age`)
) ENGINE=InnoDB AUTO_INCREMENT=70032 DEFAULT CHARSET=utf8mb3


| id | age | name |
+-------+-------+--------+
| 70031 | 80009 | n79999 |
| 70030 | 80008 | n79998 |
| 70029 | 80007 | n79997 |
| 70028 | 80006 | n79996 |
| 70027 | 80005 | n79995 |
+-------+-------+--------+

事务 1:
select * from test where age = 80009 for update;

事务 2:
select * from test where age = 80008 for update;

事务 2 会阻塞,data_locks 显示事务 2 在等待 id=70031 的 record lock ,这是为什么?
2025 次点击
所在节点    MySQL
12 条回复
hjcpnr
2023-09-24 06:00:51 +08:00
补充一下

事务 1:
select * from test where age = 80009 for update;

事务 2 ,需要阻塞等待事务 1 结束:
select * from test where age = 80008 for update;

事务 3 ,不会阻塞,成功执行:
update test set name ="haha" where age = 80008;

另外,如果把 age 的类型从 char 改成 int ,三个事务都不会互相阻塞
gumusu
2023-09-24 08:09:03 +08:00
字段类型不一致,隐式类型转换导致没走索引? explain 一下看看吧
mightybruce
2023-09-24 08:49:20 +08:00
select for update 百分百会加锁, 而且是悲观锁。
尽量少用 select for update
mightybruce
2023-09-24 08:51:30 +08:00
看 mysql 文档,了解一下间隙锁、临键锁、记录锁的区别。
rekulas
2023-09-24 08:55:46 +08:00
没走索引,字符串你要用 age = '80008'
mightybruce
2023-09-24 09:05:54 +08:00
age 在 select 中 从 char 变为 int 发生了隐式转换,这个时候不会走索引
sansui3
2023-09-24 10:51:33 +08:00
@hjcpnr 你这个有问题吧,事务 3 也会被阻塞,我在本地试了一下(我的版本是 8.0.33 );为什么被阻塞,和其他人的结论是一样的,因为字段类型不一致所以其实锁的是全表,为什么显示事务 2 在等待 id=70031 的 record lock ,就是因为事务 2 也需要按“顺序”给全表的所有数据加锁,所以在给 70027 加锁的时候就在等待了,因为事务 1 已经给 70027 这条加索了。
jorneyr
2023-09-24 13:44:30 +08:00
MySQL 的自动类型转换非常不好,遇到过几次问题都是书写不规范导致了使用自动类型转换造成的。
uselessVisitor
2023-09-24 17:15:03 +08:00
@mightybruce rc 只有行锁吧
uselessVisitor
2023-09-24 17:15:34 +08:00
看看是否走了索引,走索引就是行锁,没走的话是表锁
katsusan
2023-09-24 20:37:30 +08:00
explain 一下就能看出来 age=80009 锁了全表.

```mysql
mysql> explain select * from test where age = 80009 for update;
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+
| 1 | SIMPLE | test | NULL | ALL | idx_age | NULL | NULL | NULL | 3 | 33.33 | Using where |
+----+-------------+-------+------------+------+---------------+------+---------+------+------+----------+-------------+
1 row in set, 3 warnings (0.00 sec)

mysql> explain select * from test where age = '80009' for update;
+----+-------------+-------+------------+-------+---------------+---------+---------+-------+------+----------+-------+
| id | select_type | table | partitions | type | possible_keys | key | key_len | ref | rows | filtered | Extra |
+----+-------------+-------+------------+-------+---------------+---------+---------+-------+------+----------+-------+
| 1 | SIMPLE | test | NULL | const | idx_age | idx_age | 40 | const | 1 | 100.00 | NULL |
+----+-------------+-------+------------+-------+---------------+---------+---------+-------+------+----------+-------+
1 row in set, 1 warning (0.00 sec)

```
9fan
2023-09-25 08:53:21 +08:00
@gumusu 这位哥说的一针见血,既然都定义为 char ,还是按照规定格式来写吧,将值放在 ''

这是一个专为移动设备优化的页面(即为了让你能够在 Google 搜索结果里秒开这个页面),如果你希望参与 V2EX 社区的讨论,你可以继续到 V2EX 上打开本讨论主题的完整版本。

https://tanronggui.xyz/t/976590

V2EX 是创意工作者们的社区,是一个分享自己正在做的有趣事物、交流想法,可以遇见新朋友甚至新机会的地方。

V2EX is a community of developers, designers and creative people.

© 2021 V2EX