赞
踩
NOWAIT 和 SKIP LOCKED 是用于 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 锁定读取语句的选项,用于实现并发读取锁定。
当一个事务锁定一行时,如果另一个事务请求相同被锁定的行的 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 语句,它必须等待阻塞事务释放行锁。这种行为防止其他事务更新或删除被其他事务查询的行。然而,如果你希望在请求的行被锁定时立即返回查询结果,或者如果从结果集中排除已被锁定的行可接受的话,则不需要等待行锁释放。
为了避免等待其他事务释放行锁,可以使用 NOWAIT 和 SKIP LOCKED 选项与 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 锁定读取语句。
注意:
跳过已被锁定的行的查询返回的是数据的不一致视图。因此,SKIP LOCKED 不适用于常规的事务工作。然而,当多个会话访问同一个类似队列的表时,可以使用 SKIP LOCKED 来避免锁竞争。
NOWAIT 和 SKIP LOCKED 仅适用于行级锁。
使用 NOWAIT 或 SKIP LOCKED 的语句不适用于基于语句的复制。
以下示例演示了 NOWAIT 和 SKIP LOCKED 的用法。会话1开始一个事务,在单个记录上获取行锁。会话2尝试使用 NOWAIT 选项对相同记录进行锁定读取。由于请求的行被会话1锁定,锁定读取立即返回错误。在会话3中,使用 SKIP LOCKED 的锁定读取返回了除了会话1锁定的行之外的请求行。
- # 会话1:
-
- mysql> CREATE TABLE t (i INT, PRIMARY KEY (i)) ENGINE = InnoDB;
-
- mysql> INSERT INTO t (i) VALUES(1),(2),(3);
-
- mysql> START TRANSACTION;
-
- mysql> SELECT * FROM t WHERE i = 2 FOR UPDATE;
- +---+
- | i |
- +---+
- | 2 |
- +---+
-
- # 会话2:
-
- mysql> START TRANSACTION;
-
- mysql> SELECT * FROM t WHERE i = 2 FOR UPDATE NOWAIT;
- ERROR 3572 (HY000): Do not wait for lock.
-
- # 会话3:
-
- mysql> START TRANSACTION;
-
- mysql> SELECT * FROM t FOR UPDATE SKIP LOCKED;
- +---+
- | i |
- +---+
- | 1 |
- | 3 |
- +---+
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。