如何从PostgreSQL中的函数访问外部作用域变量?

时间:2019-01-31 10:02:49

标签: postgresql function scope plpgsql

我有此代码:

DO $$
DECLARE
    NODE_ID bigint :=  46;
BEGIN
    CREATE OR REPLACE FUNCTION funk(VAL bigint) 
    RETURNS bigint AS $f$
        BEGIN
            RETURN VAL;
        END; $f$ LANGUAGE plpgsql;

    RAISE NOTICE '%', funk(NODE_ID);
END $$;

我按预期工作,并在控制台上打印46。 我想摆脱参数,因为变量是全局的。但是我遇到了错误:

DO $$
DECLARE
    NODE_ID bigint :=  46;
BEGIN
    CREATE OR REPLACE FUNCTION funk() 
    RETURNS bigint AS $f$
        BEGIN
            RETURN NODE_ID;
        END; $f$ LANGUAGE plpgsql;

    RAISE NOTICE '%', funk();
END $$;

我收到“ NODE_ID不存在”的信息。有没有办法访问函数中的外部变量?

2 个答案:

答案 0 :(得分:1)

不,那是行不通的,因为该函数与您的DO块没有任何关系。它是一个持久性数据库对象,在DO块完成之后将继续存在于数据库中。

本质上,函数只是带有函数主体的字符串(以及一些元数据,请参见pg_proc);在这种情况下,函数主体由开始{{1}和结束$f$之间的文本组成。函数运行时由语言处理程序解释。

您只能在函数中引用的数据库数据是其他持久数据库对象,而DO块中的变量不是其中之一。

除了(在某种程度上)配置参数外,PostgreSQL中没有全局变量。您可以使用SETSHOW SQL命令来访问它们,并且更方便地通过代码使用set_configcurrent_setting函数来访问它们。

答案 1 :(得分:0)

或使用动态SQL:

DO $$
DECLARE
    NODE_ID bigint :=  46;
    src text := format('
        CREATE OR REPLACE FUNCTION funk() 
        RETURNS bigint AS $f$
            BEGIN
                RETURN %s;
            END; 
        $f$ LANGUAGE plpgsql;
    ', NODE_ID::text);
BEGIN
    execute src;
    RAISE NOTICE '%', funk();
END $$;

(为我工作,着眼于您的问题,寻找相同问题的解决方案)