在 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 中创建一个不返回复合值的函数的主要内容,如果未能解决你的问题,请参考以下文章

如何在不插入值的情况下在sql中创建动态行?

返回使用 Rocket 和 Diesel (Rust) 在 PostgreSQL 中创建的单个记录

在 Android 中创建复合视图

如果另一个系列包含特定字符串,如何在数据框中创建一个返回值的系列?

在 C# 中创建复合索引

在 postgreSQL 中创建表