锁定读取
如果查询数据,然后在同一事务中插入或更新相关数据,则常规SELECT
语句不能提供足够的保护。其他事务可以更新或删除刚查询的相同行。 InnoDB
支持两种类型的 锁定读取,这些读取提供了额外的安全性:
-
在读取的任何行上设置共享模式锁定。其他会话可以读取行,但是在事务提交之前不能修改它们。如果这些行中的任何一个被尚未提交的另一个事务更改,则查询将等待直到该事务结束,然后使用最新值。
-
对于索引记录,搜索遇到的情况,锁定行和任何关联的索引条目,就像您
UPDATE
对这些行发出 语句一样。其他事务被阻止SELECT ... LOCK IN SHARE MODE
在某些事务隔离级别更新这些行,执行操作或读取数据。一致的读取将忽略读取视图中存在的记录上设置的任何锁定。(记录的旧版本无法锁定;可以通过在记录的内存副本上应用撤消日志来重构它们 。)
这些子句在处理单个表或跨多个表的树结构或图结构数据时最有用。您从一处到另一处遍历边缘或树枝,同时保留返回的权利并更改这些“ 指针 ”值中的任何一个 。
提交或回滚事务时,将释放 由LOCK IN SHARE MODE
和 设置的所有锁定FOR UPDATE
。
注意
仅当禁用自动提交时(START TRANSACTION
通过autocommit
以0 开始事务或设置 为0 才可以进行锁定读取) 。
除非在子查询中也指定了锁定读取子句,否则外部语句中的锁定读取子句不会锁定嵌套子查询中表的行。例如,以下语句不会锁定table中的行 t2
。
SELECT * FROM t1 WHERE c1 = (SELECT c1 FROM t2) FOR UPDATE;
要锁定table中的行,t2
请向子查询添加锁定的read子句:
SELECT * FROM t1 WHERE c1 = (SELECT c1 FROM t2 FOR UPDATE) FOR UPDATE;
锁定阅读示例
假设您要在表中插入新行 child
,并确保子行在表中具有父行parent
。您的应用程序代码可以确保整个操作序列的引用完整性。
首先,使用一致的读取来查询表 PARENT
并验证父行是否存在。您可以安全地将子行插入表格 CHILD
吗?不可以,因为某些其他会话可能会在您SELECT
和您之间的时刻删除父行 INSERT
,而您却没有意识到。
为避免此潜在问题,请执行以下 SELECT
使用LOCK IN SHARE MODE
:
SELECT * FROM parent WHERE NAME = 'Jones' LOCK IN SHARE MODE;
在之后LOCK IN SHARE MODE
的查询返回父'Jones'
,你可以放心地将孩子记录添加到CHILD
表并提交事务。任何试图获取PARENT
表中适用行中的排他锁的事务都将 等到您完成操作(即所有表中的数据处于一致状态)后再进行。
再举一个例子,考虑一个表中的整数计数器字段,该字段CHILD_CODES
用于为添加到table的每个子代分配唯一标识符 CHILD
。不要使用一致读取或共享模式读取来读取计数器的当前值,因为数据库的两个用户可能会看到该计数器的相同值,并且如果两个事务尝试使用以下方式添加行,则会发生重复键错误:与CHILD
表相同的标识符。
在这里,LOCK IN SHARE MODE
这不是一个好的解决方案,因为如果两个用户同时读取计数器,则其中至少有一个在尝试更新计数器时会陷入死锁状态。
要实现读取和递增计数器,请先使用进行锁定读取FOR UPDATE
,然后再递增计数器。例如:
SELECT counter_field FROM child_codes FOR UPDATE;
UPDATE child_codes SET counter_field = counter_field + 1;
A SELECT ... FOR UPDATE
读取最新的可用数据,并在读取的每一行上设置排他锁。因此,它设置了与搜索的SQL UPDATE
在行上设置的锁相同的锁。
前面的描述只是工作方式的一个示例 SELECT ... FOR UPDATE
。在MySQL中,生成唯一标识符的特定任务实际上可以仅通过单次访问表来完成:
UPDATE child_codes SET counter_field = LAST_INSERT_ID(counter_field + 1);
SELECT LAST_INSERT_ID();
该SELECT
语句仅检索标识符信息(特定于当前连接)。它不访问任何表。
更多内容欢迎关注我的个人公众号“韩哥有话说”,100G人工智能学习资料,大量后端学习资料等你来拿。