sql - postgresql 中咨询锁超时

标签 sql oracle postgresql locking database-migration

我正在从 ORACLE 迁移。目前我正在尝试移植这个电话:

lkstat := DBMS_LOCK.REQUEST(lkhndl, DBMS_LOCK.X_MODE, lktimeout, true);

This function尝试获取锁 lkhndl,如果在 timeout 秒后未能获取锁,则返回 1。

我用的是postgresql

pg_advisory_xact_lock(lkhndl);

但是,它似乎永远在等待锁定。 pg_try_advisory_xact_lock失败立即返回。有没有办法实现锁获取的超时版本?

lock_timeout设置,但我不确定它是否适用于 advisory locks以及超时后 pg_advisory_xact_lock 的行为。

最佳答案

这是一个包装器的原型(prototype),它不能很好地模拟 DBMS_LOCK.REQUEST - 仅限于一种类型的锁(事务范围咨询锁)。

要使函数与 Oracle 完全兼容,将需要数百行代码。但这是一个开始。

CREATE OR REPLACE FUNCTION
advisory_xact_lock_request(p_key bigint, p_timeout numeric)
RETURNS integer
LANGUAGE plpgsql AS $$
/*  Imitate DBMS_LOCK.REQUEST for PostgreSQL advisory lock. 
Return 0 on Success, 1 on Timeout, 3 on Parameter Error. */
DECLARE
    t0 timestamptz := clock_timestamp();
BEGIN
    IF p_timeout NOT BETWEEN 0 AND 86400 THEN
        RAISE WARNING 'Invalid timeout parameter';
        RETURN 3;
    END IF;
    LOOP
        IF pg_try_advisory_xact_lock(key) THEN
            RETURN 0;
        ELSIF clock_timestamp() > t0 + (p_timeout||' seconds')::interval THEN
            RAISE WARNING 'Could not acquire lock in % seconds', p_timeout;
            RETURN 1;
        ELSE
            PERFORM pg_sleep(0.01); /* 10 ms */
        END IF;
    END LOOP;
END;
$$;

使用这段代码测试它:

SELECT CASE 
    WHEN advisory_xact_lock_request(1, 2.5) = 0
    THEN pg_sleep(120)
END; -- and repeat this in parallel session 

/* Usage in Pl/PgSQL */

lkstat := advisory_xact_lock_request(lkhndl, lktimeout);

关于sql - postgresql 中咨询锁超时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38853470/

相关文章:

mysql - 选择具有最大平均值的元素

database - 无法连接到 Oracle 数据库 - ORA-01033 : ORACLE initialization or shutdown in progress

mysql - 执行具有完整性约束的 'subset' 关系的最佳方法是什么

java.lang.ClassCastException : org. hibernate.action.DelayedPostInsertIdentifier 无法转换为 java.lang.Long

postgresql - Postgres 如何使 2 个数字字段彼此唯一

sql - 连接语句中的错误 "ORA-00932: inconsistent datatypes: expected - got BLOB"

sql - 无法截断表

python - Postgresql报错语句太大

sql - 在事务内部或外部打开和关闭游标以及如何在事务失败时关闭游标

c# - 如何将参数值存储在 webservice 的 oracle 过程中