PostgreSQL 中使用CTE获取时间段数据的技巧

在 PostgreSQL 中,同样可以使用 CTE(公共表表达式)来获取特定时间段的数据。PostgreSQL 提供了丰富的日期和时间函数,结合这些函数,能够轻松生成最近的月份、周、季度等时间段。下面是如何在 PostgreSQL 中获取最近十二个月、最近十二周、最近四个季度,以及十二个月前的月第一天和十二周前的周第一天的示例。

1. 获取最近十二个月

在 PostgreSQL 中,可以使用 TO_CHAR 函数结合递归CTE来生成最近十二个月的列表。

sql 复制代码
WITH RECURSIVE last_twelve_months AS (
    SELECT 
        TO_CHAR(CURRENT_DATE, 'YYYY-MM') AS year_month,
        CURRENT_DATE::DATE AS date_value
    UNION ALL
    SELECT 
        TO_CHAR(date_value - INTERVAL '1 MONTH', 'YYYY-MM'),
        (date_value - INTERVAL '1 MONTH')::DATE
    FROM last_twelve_months
    WHERE date_value > CURRENT_DATE - INTERVAL '11 MONTH'
)
SELECT year_month
FROM last_twelve_months
ORDER BY year_month DESC;

获取最近十二周

sql 复制代码
WITH RECURSIVE last_twelve_weeks AS (
    SELECT 
        TO_CHAR(DATE_TRUNC('week', CURRENT_DATE), 'YYYY-MM-DD') AS week_start,
        DATE_TRUNC('week', CURRENT_DATE)::DATE AS date_value
    UNION ALL
    SELECT 
        TO_CHAR(DATE_TRUNC('week', date_value - INTERVAL '1 WEEK'), 'YYYY-MM-DD'),
        DATE_TRUNC('week', date_value - INTERVAL '1 WEEK')::DATE
    FROM last_twelve_weeks
    WHERE date_value > CURRENT_DATE - INTERVAL '11 WEEK'
)
SELECT week_start
FROM last_twelve_weeks
ORDER BY week_start DESC;

获取最近四个季度

sql 复制代码
WITH RECURSIVE last_four_quarters AS (
    SELECT 
        EXTRACT(YEAR FROM CURRENT_DATE) AS year,
        EXTRACT(QUARTER FROM CURRENT_DATE) AS quarter,
        CURRENT_DATE::DATE AS date_value
    UNION ALL
    SELECT 
        CASE WHEN quarter = 1 THEN year - 1 ELSE year END,
        CASE WHEN quarter = 1 THEN 4 ELSE quarter - 1 END,
        (date_value - INTERVAL '3 MONTH')::DATE
    FROM last_four_quarters
    WHERE date_value > CURRENT_DATE - INTERVAL '1 YEAR'
)
SELECT CONCAT(year, '-Q', quarter) AS year_quarter
FROM last_four_quarters
ORDER BY year DESC, quarter DESC;
  1. 获取十二个月前的月第一天
sql 复制代码
SELECT DATE_TRUNC('month', CURRENT_DATE - INTERVAL '12 MONTH') AS first_day_of_month;
  1. 获取十二周前的周第一天
sql 复制代码
SELECT DATE_TRUNC('week', CURRENT_DATE - INTERVAL '12 WEEK') AS first_day_of_week;

测试和调试

  • 确保在执行这些查询之前,检查 PostgreSQL 的版本,因为有些功能(如递归 CTE)在非常旧的版本中可能不支持。
  • 如果仍然无法正确获取数据,可以逐步调试,如单独运行每一部分的查询,检查 CURRENT_DATEINTERVAL 的使用是否正确。
  • 还可以直接使用 NOW() 替代 CURRENT_DATE 来确保取到的是完整的时间戳数据。
相关推荐
大布布将军4 小时前
⚡️ 深入数据之海:SQL 基础与 ORM 的应用
前端·数据库·经验分享·sql·程序人生·面试·改行学it
JIngJaneIL5 小时前
基于java+ vue农产投入线上管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot
源代码•宸5 小时前
goframe框架签到系统项目(BITFIELD 命令详解、Redis Key 设计、goframe 框架教程、安装MySQL)
开发语言·数据库·经验分享·redis·后端·mysql·golang
川贝枇杷膏cbppg5 小时前
Redis 的 AOF
java·数据库·redis
TG:@yunlaoda360 云老大6 小时前
如何在华为云国际站代理商控制台进行SFS Turbo的性能与容量核查?
服务器·网络·数据库·华为云
ytttr8736 小时前
MATLAB基于LDA的人脸识别算法实现(ORL数据库)
数据库·算法·matlab
云老大TG:@yunlaoda3607 小时前
如何进行华为云国际站代理商跨Region适配?
大数据·数据库·华为云·负载均衡
思成不止于此7 小时前
【MySQL 零基础入门】事务精讲(二):ACID 特性与并发问题
数据库·笔记·学习·mysql
Boilermaker19927 小时前
[MySQL] 初识 MySQL 与 SQL 基础
数据库·mysql
今晚务必早点睡7 小时前
Redis——快速入门第二课:Redis 常用命令 + 能解决实际问题
数据库·redis·bootstrap