1

我面临类似这样的查询:

SELECT *
FROM invoicable_interval i,
  LATERAL partition_into_months(i.start_date, i.stop_or_current_date) p;

... 其中 partition_into_months 的定义类似于:

CREATE FUNCTION partition_into_months(start_date DATE, stop_date DATE)
  RETURNS TABLE (start_date DATE, stop_date DATE, days INT)
AS $$ ... $$
LANGUAGE sql
IMMUTABLE;

因此,我正在为辅助表使用可变间隔进行交叉连接,因此使用(冗余)关键字 LATERAL。

这适用于 PostgreSQL 9.3.6 和 9.4.2,但是,对于 9.1.3,我收到以下错误消息:

[42601] ERROR: syntax error at or near "."
  Position: 398

指示的位置是表达式“i.start_date”中的点。

如何重写此 SQL 查询以将其移植回 PostgreSQL 9.1.3 ?

4

1 回答 1

1

SELECTPostgreSQL 支持在子句中调用集合返回函数。现在我们已经不赞成这样做LATERAL并且肯定不鼓励这样做,因为它的行为相当不稳定,但它仍然有用。

在你的情况下,你可以写:

SELECT 
  i.*,
  (partition_into_months(i.start_date, i.stop_or_current_date)).*
FROM invoicable_interval i;

但是,这可能会导致对partition_into_months每列返回一次调用,因为(fn).*基本上是宏扩展为(fn).col1, (fn).col2, .... 为避免这种情况,您可以将其包装在子查询中,例如

SELECT (x.i).*, (x.p).*
FROM
(
  SELECT 
    i,
    partition_into_months(i.start_date, i.stop_or_current_date) p
  FROM invoicable_interval i
) x(i,p);

请注意,如果列表中存在多个集合返回函数,则会遇到奇怪的结果SELECT。这不是您所期望的交叉连接。例如,比较:

SELECT generate_series(1,4), generate_series(1,4)

SELECT generate_series(1,4), generate_series(1,3);
于 2015-05-29T01:27:53.333 回答