当前位置:   article > 正文

mysql中使用 NOWAIT 和 SKIP LOCKED 实现锁定读取并发性

mysql中使用 NOWAIT 和 SKIP LOCKED 实现锁定读取并发性

NOWAIT 和 SKIP LOCKED 是用于 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 锁定读取语句的选项,用于实现并发读取锁定。

当一个事务锁定一行时,如果另一个事务请求相同被锁定的行的 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 语句,它必须等待阻塞事务释放行锁。这种行为防止其他事务更新或删除被其他事务查询的行。然而,如果你希望在请求的行被锁定时立即返回查询结果,或者如果从结果集中排除已被锁定的行可接受的话,则不需要等待行锁释放。

为了避免等待其他事务释放行锁,可以使用 NOWAIT 和 SKIP LOCKED 选项与 SELECT ... FOR UPDATE 或 SELECT ... FOR SHARE 锁定读取语句。

  • NOWAIT:使用 NOWAIT 的锁定读取永远不会等待获取行锁。该查询立即执行,如果请求的行被锁定,则失败并返回错误信息。
  • SKIP LOCKED:使用 SKIP LOCKED 的锁定读取永远不会等待获取行锁。该查询立即执行,从结果集中移除已被锁定的行。

注意:

跳过已被锁定的行的查询返回的是数据的不一致视图。因此,SKIP LOCKED 不适用于常规的事务工作。然而,当多个会话访问同一个类似队列的表时,可以使用 SKIP LOCKED 来避免锁竞争。

NOWAIT 和 SKIP LOCKED 仅适用于行级锁

使用 NOWAIT 或 SKIP LOCKED 的语句不适用于基于语句的复制。

以下示例演示了 NOWAIT 和 SKIP LOCKED 的用法。会话1开始一个事务,在单个记录上获取行锁。会话2尝试使用 NOWAIT 选项对相同记录进行锁定读取。由于请求的行被会话1锁定,锁定读取立即返回错误。在会话3中,使用 SKIP LOCKED 的锁定读取返回了除了会话1锁定的行之外的请求行。

  1. # 会话1:
  2. mysql> CREATE TABLE t (i INT, PRIMARY KEY (i)) ENGINE = InnoDB;
  3. mysql> INSERT INTO t (i) VALUES(1),(2),(3);
  4. mysql> START TRANSACTION;
  5. mysql> SELECT * FROM t WHERE i = 2 FOR UPDATE;
  6. +---+
  7. | i |
  8. +---+
  9. | 2 |
  10. +---+
  11. # 会话2:
  12. mysql> START TRANSACTION;
  13. mysql> SELECT * FROM t WHERE i = 2 FOR UPDATE NOWAIT;
  14. ERROR 3572 (HY000): Do not wait for lock.
  15. # 会话3:
  16. mysql> START TRANSACTION;
  17. mysql> SELECT * FROM t FOR UPDATE SKIP LOCKED;
  18. +---+
  19. | i |
  20. +---+
  21. | 1 |
  22. | 3 |
  23. +---+

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/IT小白/article/detail/193039
推荐阅读
相关标签
  

闽ICP备14008679号