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锁定,锁定读取立即返回