4

我想将多个查询组合成一个存在于 PostgreSQL 中的函数。该函数将使用 PDO 进行查询。

功能是:

CREATE OR REPLACE FUNCTION "test_multipe_refcursor"() 
RETURNS SETOF refcursor AS $BODY$

DECLARE
    parentRC refcursor;
    childRC refcursor;

BEGIN

open parentRC FOR
SELECT * FROM parent;
RETURN NEXT parentRC;

open childRC FOR
SELECT * FROM child;
RETURN NEXT childRC;

RETURN;

END;$BODY$
LANGUAGE 'plpgsql' VOLATILE;

ALTER FUNCTION "test_multipe_refcursor"() OWNER TO postgres;

这是PHP代码。“数据库”作为设置常用连接属性的单例类,没什么特别的。

  $database = Database::load();
  $sql = "select * from test_multipe_refcursor();";
  $p = $database->query($sql);

  $i = 1;
  do
  {
     $this->set('set' . $i, $p->fetchAll(PDO::FETCH_ASSOC));
     $i++;
  } while ($p->nextRowset());

  $p->closeCursor();

和结果。

 PDOException: SQLSTATE[IM001]: Driver does not support this function: driver does not support multiple rowsets in xxxx.php on line 32

这似乎表明它不受支持,但话又说回来,我找不到确切定义什么的列表。

有没有人设法让这个工作?

参考:

4

2 回答 2

4

对返回多个结果集的支持仍在PostgreSQL 待办事项列表中,并且肯定不会达到 8.4。至于 setof refcursors 方法,您尝试执行的操作不起作用,因为该函数没有返回多个行集 - 它返回一个行集的 refcursors。我不确定使用 refcursors 客户端是否有效,但我认为它不太可能,即使客户端-服务器协议支持它,PDO 也不太可能为此提供 API。

但是你为什么要在一个查询中返回多个结果集呢?您始终可以单独进行查询。

于 2009-06-06T00:58:21.960 回答
1

在这个 PostgreSQL 文档页面的底部附近,有一节描述了如何从函数中传回一个或多个游标。基本上,您让调用者指定游标的名称作为参数:

CREATE FUNCTION myfunc(refcursor, refcursor) RETURNS SETOF refcursor AS $$
BEGIN
    OPEN $1 FOR SELECT * FROM table_1;
    RETURN NEXT $1;
    OPEN $2 FOR SELECT * FROM table_2;
    RETURN NEXT $2;
END;
$$ LANGUAGE plpgsql;

-- need to be in a transaction to use cursors.
BEGIN;

SELECT * FROM myfunc('a', 'b');

FETCH ALL FROM a;
FETCH ALL FROM b;
COMMIT;

该页面适用于 PostgreSQL 8.4,但此文档片段至少可以追溯到 8.1(我正在运行的版本)。正如评论所说,您需要在事务内部才能使用游标,因为它们在每个事务结束时隐式关闭(即,如果自动提交模式打开,则在每个语句的末尾)。

于 2009-10-03T14:20:04.337 回答