postgresql 中的咨询锁超时

Posted

技术标签:

【中文标题】postgresql 中的咨询锁超时【英文标题】:Timeout on advisory locks in postgresql 【发布时间】:2016-12-15 15:51:45 【问题描述】:

我正在从 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 在超时后的行为。

【问题讨论】:

你可以设置一个statement_timeout,比如SET LOCAL。不幸的是,这是会话(或LOCAL,事务)级别,而不是语句级别。 我没关系。但是超时后会发生什么?我如何知道是否获得了锁? 超时后,执行被中止,我认为您甚至无法捕获这些错误。为了完全模拟那个 Oracle 函数,恐怕你需要编写一个重试循环,可能有 pg_sleep() 调用(但是我不确定它是如何执行的,我从来没有写过这样的东西)。 哦,我担心会是这样的:( 老实说,我们似乎真的可以使用补丁来实现类似于您现在在 Oracle 中使用的功能的功能。你的 C 编程怎么样? 【参考方案1】:

这是一个包装器的原型,它对 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);

【讨论】:

以上是关于postgresql 中的咨询锁超时的主要内容,如果未能解决你的问题,请参考以下文章

PostgreSQL 咨询锁不适用于 Doctrine 的 DBAL

PostgreSQL 锁监控

使用 node-postgres 监听查询超时?

如何从应用程序级别设置 postgres 查询执行超时?

节点 PostgreSQL 超时客户端查询

PostgreSQL 9.4.2 中的“CREATE TRIGGER”使用啥锁(如果有)