当前位置: 首页 > news >正文

掌握 CTE 技巧,实现连续日期和月份的 SQL 报表统计

在 SQL 查询中,报表统计往往涉及到特定时间段内的数据汇总,如每日、每月的销售数据等。然而,面对缺少数据的日期或月份,传统 SQL 查询可能会直接跳过这些日期,使得输出的报表在视觉上并不连续。本文将展示如何利用 CTE(通用表表达式)生成完整的时间范围,并确保报表统计结果中,数据缺失的日期或月份以 0 填充。通过这种方法,可以大大提高数据报表的完整性,使结果更加连贯清晰。

需求背景

假设我们在电商系统中有一个订单表 orders,其结构如下:

  • order_id:订单编号
  • order_date:订单日期
  • amount:订单金额

我们将基于该表,构建以下几种常见的报表统计需求:

  1. 最近 30 天的每日销售统计,包含数据缺失的日期。
  2. 最近 6 个月的月度销售统计,包含没有数据的月份。

在这两个场景中,我们希望输出的结果表中,不论是否有数据,特定时间段内的每一天或每一月都应该显示,并且缺失数据的日期或月份的销售额显示为 0


示例 1:最近 30 天的每日销售统计

为了显示最近 30 天的每日销售数据,并确保每一天都显示出来,我们首先可以利用递归 CTE 生成一个完整的 30 天日期范围,然后通过 LEFT JOIN 将订单数据连接上去。

SQL 查询示例

-- 递归 CTE 生成最近 30 天的日期范围
WITH RECURSIVE DateRange AS (SELECT CURDATE() - INTERVAL 29 DAY AS dateUNION ALLSELECT date + INTERVAL 1 DAYFROM DateRangeWHERE date + INTERVAL 1 DAY <= CURDATE()
),
Last30Days AS (SELECT order_date, amountFROM ordersWHERE order_date >= CURDATE() - INTERVAL 30 DAY
)
SELECT d.date AS order_date,COALESCE(SUM(l.amount), 0) AS daily_sales
FROM DateRange d
LEFT JOIN Last30Days l ON d.date = l.order_date
GROUP BY d.date
ORDER BY d.date;

查询解析

  1. DateRange CTE:生成最近 30 天的完整日期范围。
  2. Last30Days CTE:筛选出订单表中最近 30 天的数据。
  3. 主查询:通过 LEFT JOINDateRangeLast30Days 连接在一起,确保每一天都出现在结果中。使用 COALESCE 函数将没有数据的日期销售额填充为 0

通过该查询,我们可以得到一个包含最近 30 天每日销售额的表格,其中没有订单数据的日期也会显示为 0

示例 2:最近 6 个月的月度销售统计

同样地,为了展示最近 6 个月的月度销售数据,并包含没有订单的月份,我们可以生成一个完整的 6 个月月份范围,再将订单数据连接上去。

SQL 查询示例

-- 递归 CTE 生成最近 6 个月的月份范围
WITH RECURSIVE MonthRange AS (SELECT DATE_FORMAT(CURDATE() - INTERVAL 5 MONTH, '%Y-%m') AS monthUNION ALLSELECT DATE_FORMAT(DATE_ADD(STR_TO_DATE(month, '%Y-%m'), INTERVAL 1 MONTH), '%Y-%m')FROM MonthRangeWHERE DATE_ADD(STR_TO_DATE(month, '%Y-%m'), INTERVAL 1 MONTH) <= CURDATE()
),
Last6Months AS (SELECT DATE_FORMAT(order_date, '%Y-%m') AS month, amountFROM ordersWHERE order_date >= CURDATE() - INTERVAL 6 MONTH
)
SELECT m.month,COALESCE(SUM(l.amount), 0) AS monthly_sales
FROM MonthRange m
LEFT JOIN Last6Months l ON m.month = l.month
GROUP BY m.month
ORDER BY m.month;

查询解析

  1. MonthRange CTE:通过递归 CTE 生成最近 6 个月的完整月份范围。
  2. Last6Months CTE:提取订单表中最近 6 个月的订单数据,并格式化日期为 YYYY-MM 月份格式。
  3. 主查询:利用 LEFT JOINMonthRangeLast6Months 连接在一起,确保每个月都出现在最终结果中。使用 COALESCE 确保无数据的月份销售额为 0

这样,我们可以得到包含最近 6 个月每月销售额的表格,其中没有订单数据的月份会显示为 0,保证了数据的连续性。


总结

通过上述两个示例,我们可以看到 CTE 的强大之处。借助递归 CTE,我们可以轻松生成日期或月份范围,并将其与实际数据进行连接,确保报表统计结果的连续性。这种方法尤其适用于时间跨度较大的报表查询场景,如日统计、月统计等。

CTE 技术不仅帮助我们提高了 SQL 代码的可读性,也为实现更完整的报表数据提供了便捷手段。无论是日统计还是月统计,通过 CTE,我们都可以确保报表结果具有更高的业务价值。希望本文能帮助你掌握 CTE 的使用技巧,并应用于实际的报表开发中,为业务分析提供更精确的数据支持。

http://www.lryc.cn/news/475157.html

相关文章:

  • 【表格解决问题】EXCEL行数过多,WPS如何按逐行分别打印多个纸张中
  • Maven讲解从基础到高级配置与实践
  • Vue3组件式父子传值
  • 网页自动化测试和爬虫:Selenium库入门与进阶
  • Cells 单元
  • 2024/11/2 安卓创建首页界面
  • SpringSession源码分析
  • IIC
  • LLM Observability: Azure OpenAI (一)
  • qt QBrush详解
  • Excel函数CUnique连接合并指定区域的唯一值
  • 机械革命屏幕设置为RGB
  • 开源项目-投票管理系统
  • LeetCode 104.二叉树的最大深度
  • Android启动流程_Init阶段
  • 萤火虫算法优化BILSTM神经网络多输入回归分析
  • 在线QP(QuotedPrintable)编码解码工具
  • 【已解决】cra 配置路径别名 @ 后,出现 ts 报错:找不到模块“@/App”或其相应的类型声明。ts(2307)
  • leetcode-643. 子数组最大平均数 I
  • 论分布式架构设计及其实现
  • 基于BP神经网络的手写体数字图像识别
  • QT——串口调试助手
  • 国产操作系统卖疯了!最营收7.84亿,最低1.5亿
  • 2024年华为OD机试真题-最小的调整次数-Python-OD统一考试(E卷)
  • React.js教程:从JSX到Redux的全面解析
  • 二叉苹果树
  • 【大数据学习 | kafka】producer的参数与结构
  • 2. 从服务器的主接口入手
  • nginx上传文件超过限制大小、响应超时、反向代理请求超时等问题解决
  • 第16课 核心函数(方法)