按天统计数据,如果当天没有记录,则统计为0

本文介绍了一种在MySQL中实现按天统计特定表记录数的方法,并通过左连接(left join)来填充缺失日期的技巧,确保统计结果连续且完整。
今天做一个统计需求,按天统计这张表的记录数,按照pay_month和格式化之后的payment_time来分组,这张表大致数据如下
[img]http://dl2.iteye.com/upload/attachment/0114/5055/524040d7-c44c-3575-af7f-ab95ed67b85a.png[/img]

按照pay_month和格式化之后的payment_time分组之后,数据如下


[img]http://dl2.iteye.com/upload/attachment/0114/5059/fec56638-7a73-352a-bd8c-3d0f9693a0b5.png[/img]

我需要统计的是10天前到今天的数据,但就pay_month=1的数据来讲,只有2,3,4,7号有数据。而我想要没有数据的那一天同样统计出来并用0来填充。
第一反应是用left join,但由于mysql没有oracle的connect by level,只能用一张数据条数绝对大于10的表来代替select level from dual connect by level < 10;我就选了tb_order这张表本身
SELECT @cdate := date_add(@cdate,interval -1 day) days from 
(SELECT @cdate := CURDATE() from tb_order limit 10) t1


[img]http://dl2.iteye.com/upload/attachment/0114/5061/9265c8ff-416d-3514-a2bc-fc9268bd5677.png[/img]
再left join需要分组的表
select t2.days,t3.begin_day,t3.end_day,t3.pay_month,
t3.pay_day,IFNULL(t3.score,0) total_score from
(SELECT @cdate := date_add(@cdate,interval -1 day) days from
(SELECT @cdate := CURDATE() from tb_order limit 10) t1) t2
LEFT JOIN
(select CURDATE() end_day,PAY_MONTH,
date_add(CURDATE(),interval -10 day) begin_day,
date_format(PAYMENT_TIME,'%Y-%m-%d') pay_day,
count(1) score from tb_order
where ORDER_STATUS='ORDER_USED'
and CURDATE()<=date_add(date_format(PAYMENT_TIME,'%Y-%m-%d'),interval 10 day)
group by PAY_MONTH,date_format(PAYMENT_TIME,'%Y-%m-%d')) t3
on t2.days=t3.pay_day

最终达到的结果如下:

[img]http://dl2.iteye.com/upload/attachment/0114/5065/7f12145e-6eb6-3048-a894-e5f2806b1e66.png[/img]

正真需要的只有days和total_score两列,这样就算是完成了。
在 SQL 中实现按统计数据并在没有数据的日期填充 `0`,通常需要以下几个步骤: 1. **生成连续日期范围**:创建一个包含目标时间段内所有日期的结果集。 2. **左连接原始数据**:将生成的日期结果与实际数据进行左连接。 3. **使用 COALESCE 填充空值**:将没有匹配记录统计字段设置为 `0`。 以下是一个完整的 SQL 示例,假设有一个名为 `sales` 的表,其中包含字段 `sale_date`(销售日期) `amount`(销售金额),我们希望按统计销售额,并对没有销售记录的日期返回 `0`。 ### 示例 SQL 查询 ```sql -- 生成指定日期范围内的连续日期 WITH RECURSIVE DateRange AS ( SELECT '2023-01-01' AS date UNION ALL SELECT DATE_ADD(date, INTERVAL 1 DAY) FROM DateRange WHERE date < '2023-01-31' ), -- 按日期分组统计销售额 SalesSummary AS ( SELECT sale_date, SUM(amount) AS total_amount FROM sales WHERE sale_date BETWEEN '2023-01-01' AND '2023-01-31' GROUP BY sale_date ) -- 将连续日期与实际销售数据进行左连接,并用 COALESCE 填充空值 SELECT dr.date, COALESCE(ss.total_amount, 0) AS total_amount FROM DateRange dr LEFT JOIN SalesSummary ss ON dr.date = ss.sale_date ORDER BY dr.date; ``` ### 解释 - **DateRange CTE**:使用递归查询生成从 `'2023-01-01'` 到 `'2023-01-31'` 的连续日期。 - **SalesSummary CTE**:对 `sales` 表中指定时间范围内的数据进行分组统计。 - **LEFT JOIN**:将生成的日期列表与实际统计数据进行左连接,确保即使某一没有销售记录,也会出现在最终结果中。 - **COALESCE**:用于替换 `NULL` 值为 `0`,表示当天没有销售发生。 ### 注意事项 - 如果数据库不支持递归查询(如 MySQL 5.x),可以使用其他方式生成日期范围,例如通过数字表或存储过程。 - 对于 MySQL 8.0 及以上版本,可以使用 `WITH RECURSIVE` 来生成日期范围[^2]。 - 在生产环境中,应根据实际情况调整日期范围查询性能优化策略。 ### 使用变量生成连续日期(适用于 MySQL) ```sql SELECT (@date := DATE_ADD(@date, INTERVAL 1 DAY)) AS dates, COALESCE(SUM(s.amount), 0) AS total_amount FROM (SELECT @date := '2022-09-30') temp, (SELECT 1 UNION ALL SELECT 2 UNION ALL SELECT 3 UNION ALL SELECT 4 UNION ALL SELECT 5) numbers, sales s WHERE @date < '2022-10-31' GROUP BY dates; ``` 此方法利用了变量 `@date` 一个虚拟的 `numbers` 表来生成指定范围内的连续日期,并结合 `COALESCE` 实现缺失日期的数据填充。 ---
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值