在 Postgresql 中创建一个不返回复合值的函数
Posted
技术标签:
【中文标题】在 Postgresql 中创建一个不返回复合值的函数【英文标题】:Creating a function in Postgresql that does not return composite values 【发布时间】:2012-12-10 23:02:41 【问题描述】:我正在学习如何在 Postgresql 中编写函数。我定义了一个名为_tmp_myfunction()
的函数,它接受id
并返回一个表(我还定义了一个名为_tmp_mytable
的表对象类型)
-- create object type to be returned
CREATE TYPE _tmp_mytable AS (
id integer,
cost double precision
);
-- create function which returns query
CREATE OR REPLACE FUNCTION _tmp_myfunction(
id integer
)
RETURNS SETOF _tmp_mytable AS $$
BEGIN
RETURN QUERY
SELECT
sales.gid,
cost
FROM
sales
WHERE
id = sales.gid;
END;
$$ LANGUAGE plpgsql;
当我使用 id
并使用以下方法调用它时,这很好用:
SELECT * FROM _tmp_myfunction(402);
我想做的是调用它,但使用一列值而不是一个值。但是,如果我使用以下方法,我最终会将表的所有值放在一列中,用逗号分隔:
-- call function using all values in a column
SELECT _tmp_myfunction(t.id)
FROM transactions as t;
我知道如果我使用 SELECT _tmp_myfunction(402);
而不是 SELECT * FROM _tmp_myfunction(402);
可以获得相同的结果,但我不知道如何构造我的函数,这样当我传入值列。
【问题讨论】:
刚刚更新了我的变量名称以使其更清晰一些(我已更改它们以使我的问题更清晰,但错过了会有冲突)。 @ErwinBrandstetter 虽然您可以在SELECT
中调用集合返回函数,但这种行为确实很奇怪。这是一个遗留的 PostgreSQL 特定 hack,一旦 PostgreSQL 9.3 的 LATERAL
可用,就应该放弃它。至于有多奇怪,请将 SELECT generate_series(1,3), generate_series(1,3);
与 SELECT generate_series(1,3), generate_series(1,4);
进行比较。第一个返回三行,成对的结果按顺序排列。第二个返回 12 行,所有可能的结果配对,如叉积。
【参考方案1】:
你可以这样写:
SELECT (t2.function_row).id,
(t2.function_row).cost
FROM (SELECT _tmp_myfunction(t.id) as function_row
FROM transactions t ) t2;
它将为您提供字段,而不是复合行。
【讨论】:
请注意,PostgreSQL 可能会为每一行重复执行该函数,每个(func_row).column
执行一次。在函数中添加RAISE NOTICE
以了解我的意思。 9.3 的LATERAL
支持应该最终解决这个问题。有一篇包含详细信息和示例的 depesz 文章,我目前似乎找不到。
@Craig: You probably mean this one. 也期待这个功能。以上是关于在 Postgresql 中创建一个不返回复合值的函数的主要内容,如果未能解决你的问题,请参考以下文章
返回使用 Rocket 和 Diesel (Rust) 在 PostgreSQL 中创建的单个记录