在PostgreSQL中的2个日期之间每月创建一行

时间:2016-12-19 22:27:24

标签: postgresql postgresql-9.3 date-range generate-series

我需要在表格中每个人的2个日期之间每月创建一行(结果应该是该月的第一天)。例如,如果我的源表中有以下数据:

rowID | person      | startdate   | enddate
1     | 12345       | 2014-04-01  | 2014-11-30
2     | 67890       | 2014-03-01  | 2014-05-01

我希望目标表中的结果为:

person | month
12345  | 2014-04-01
12345  | 2014-05-01
12345  | 2014-06-01
12345  | 2014-07-01
12345  | 2014-08-01
12345  | 2014-09-01
12345  | 2014-10-01
12345  | 2014-11-01
67890  | 2014-03-01
67890  | 2014-04-01
67890  | 2014-05-01

非常感谢您的帮助。

2 个答案:

答案 0 :(得分:4)

无需CTE或横向连接:

$this->input->post('myname')

答案 1 :(得分:1)

计算每个月的第一天的每个人的最短和最长日期,然后使用generate_series生成这些日期之间的基于月度的范围:

WITH date_ranges AS (
SELECT 
  person,
  min(date_trunc('month', startdate))::timestamptz AS min_start,
  max(date_trunc('month', enddate))::timestamptz AS max_end
FROM person_table
GROUP BY 1
)
SELECT 
  dr.person,
  ser.month::DATE as month
FROM date_ranges AS dr,
     generate_series(min_start, max_end, '1 month') AS ser(month)

<强>输出

 person |   month
--------+------------
  12345 | 2014-04-01
  12345 | 2014-05-01
  12345 | 2014-06-01
  12345 | 2014-07-01
  12345 | 2014-08-01
  12345 | 2014-09-01
  12345 | 2014-10-01
  12345 | 2014-11-01
  67890 | 2014-03-01
  67890 | 2014-04-01
  67890 | 2014-05-01

它是如何工作的?函数调用的隐式LATERAL JOIN强制计算输入中的每一行。

此解决方案考虑到每个人可能拥有超过1行的日期,并且它具有尽可能高的范围。