9

有没有一种方法可以测试一行是否已在 Oracle 中被锁定以进行更新?

例如,假设由一位用户执行以下查询:

select * from SOME_TABLE where THE_ID = 1000 for update;

对于另一个用户,我想检查该行THE_ID = 1000是否被锁定。如果我尝试更新或其他操作,第二个用户会被阻止并继续等待(不希望这样)。

我还尝试使用第二个用户运行以下查询:

select * from SOME_TABLE where THE_ID = 1000 for update NOWAIT;

由于我不能在同一行放置两个锁,这将失败。它确实如此。我收到“ORA-00054:资源繁忙,并以 NOWAIT 指定错误获取”。我是否可以始终依靠此错误来检查锁的存在,或者是否有一种更简单、更简洁的方法来确定一行是否被锁定?

谢谢!

4

2 回答 2

16

您可以使用 FOR UPDATE NOWAIT 编写一个过程,并在行被锁定时返回错误消息:

SQL> CREATE OR REPLACE PROCEDURE do_something(p_id NUMBER) IS
  2     row_locked EXCEPTION;
  3     PRAGMA EXCEPTION_INIT(row_locked, -54);
  4  BEGIN
  5     FOR cc IN (SELECT *
  6                  FROM some_table
  7                 WHERE ID = p_id FOR UPDATE NOWAIT) LOOP
  8        -- proceed with what you want to do;
  9        NULL;
 10     END LOOP;
 11  EXCEPTION
 12     WHEN row_locked THEN
 13        raise_application_error(-20001, 'this row is locked...');
 14  END do_something;
 15  /

Procedure created

现在让我们构建一个包含两个会话的小示例:

session_1> select id from some_table where id = 1 for update;

        ID
----------
         1

session_2> exec do_something(1);

begin do_something(1); end;

ORA-20001: this row is locked...
ORA-06512: at "VNZ.DO_SOMETHING", line 11
ORA-06512: at line 2

session_1> commit;

Commit complete

session_2> exec do_something(1);

PL/SQL procedure successfully completed
于 2009-10-15T12:46:23.333 回答
1

它既不简单也不干净,但信息在V$LOCKandV$SESSION视图中可用。

但是,如果您觉得需要在您的正常应用程序代码中使用这样的内容,您需要重新考虑。应用程序不应该关心数据库如何锁定。如果您遇到死锁,您需要重组您的查询,以便它们不会发生。

于 2009-10-15T12:34:31.967 回答