在美团的面试过程中,关于数据库的提问尤其频繁,考官通常会要求考生手动编写SQL语句,并询问这些语句所加的锁。与其他公司相比,这种考察方式相对少见,许多求职者在面对这一问题时常常感到困惑。本文将详细总结InnoDB存储引擎中行锁的加锁规则,并通过实例进行说明。
InnoDB存储引擎的行锁类型简介
InnoDB存储引擎主要有三种行锁类型:
- 记录锁(Record Lock):对单一行记录进行锁定。
- 间隙锁(Gap Lock):锁定一段区间,左开右闭。
- 临键锁(Next-key Lock):锁定一段区间,左开右闭。
何种SQL语句会加锁?
- 对于常用的DML语句(例如
UPDATE
、DELETE
和INSERT
),InnoDB会自动为相关记录行加写锁。 - 默认情况下,普通的
SELECT
语句不会加锁,但在可串行化隔离级别下会加行级读锁。
上述两种情况属于隐式锁定,此外,InnoDB还支持通过特定语句进行显式锁定:
- 使用
SELECT * FROM table_name WHERE ... FOR UPDATE
会加行级写锁。 - 使用
SELECT * FROM table_name WHERE ... LOCK IN SHARE MODE
会加行级读锁。
在深入学习行锁加锁规则之前,考生需牢记以下两条核心原则:
- 只有在查找过程中访问到的对象才会加锁。
- 加锁的基本单位是Next-key Lock。
SQL语句示例分析
为了更好地理解这些加锁规则,我们将分析一条SQL语句在InnoDB中被自动加的锁。
假设有一张 user
表,其中 id
是主键(唯一索引),a
是普通索引(非唯一索引),b
是没有任何索引的普通列:
id (唯一索引) | a (非唯一索引) | b |
---|---|---|
10 | 4 | Alice |
15 | 8 | Bob |
20 | 16 | Cilly |
25 | 32 | Druid |
30 | 64 | Erik |
案例 1:唯一索引等值查询
-
记录存在的情况
查询语句如下:SELECT * FROM user WHERE id = 25 FOR UPDATE;
由于查询的记录存在,Next-key Lock将退化为记录锁,最终加锁范围是
id = 25
这一行。 -
记录不存在的情况
查询语句如下:SELECT * FROM user WHERE id = 22 FOR UPDATE;
由于
id = 22
的记录不存在,Next-key Lock将退化为间隙锁,最终加锁范围为(20, 25)
。
案例 2:唯一索引范围查询
查询语句如下:
SELECT * FROM user WHERE id >= 20 AND id < 22 FOR UPDATE;
这个查询首先会加锁范围为(15, 20]
,由于id = 20
存在,因此加锁退化为记录锁。然后继续查找,直到找到不满足条件的记录id = 25
,最终在主键id上的加锁范围为Record Lock id = 20
和 Gap Lock (20, 25)
。
案例 3:非唯一索引等值查询
-
记录存在的情况
查询语句如下:SELECT * FROM user WHERE a = 16 FOR UPDATE;
将加锁范围为Next-key Lock
(8, 16]
和Gap Lock(16, 32)
,最终结果相应。 -
记录不存在的情况
查询语句如下:SELECT * FROM user WHERE a = 18 FOR UPDATE;
由于
a = 18
的记录不存在,因此 Next-key Lock 退化为间隙锁,最终加锁范围为(16, 32)
。
案例 4:非唯一索引范围查询
查询语句如下:
SELECT * FROM user WHERE a >= 16 AND a < 18 FOR UPDATE;
首先加锁范围为(8, 16]
,然后继续查找到不满足条件的记录a = 32
,最终加锁范围为 Next-key Lock (8, 16]
和 (16, 32]
,也就是 (8, 32]
。
总结
本文深入探讨了美团面试中SQL语句加锁的细节,理解InnoDB存储引擎的加锁规则不仅可以帮助求职者在面试中应对相关问题,亦能提升对数据库并发控制机制的理解。希望这些实例分析能帮助各位在未来的面试中取得成功。