MySQL按日分组须用DATE(created_at)或DATE_FORMAT(created_at,'%Y-%m-%d'),周分组推荐YEARWEEK(created_at,1),月分组用DATE_FORMAT(created_at,'%Y-%m');空日期需在PHP预生成或MySQL生成序列补全,图表数据须保证labels与data顺序严格对应且时区统一。

用 date_format() 或 DATE() 按日分组最稳
MySQL 中时间字段是 DATETIME 或 TIMESTAMP 类型时,直接 GROUP BY created_at 会按秒级分组,根本不是“按日”。必须先截断到日期粒度:
— MySQL 用 DATE(created_at)(推荐,语义清晰、索引友好)
— 或 date_format(created_at, '%Y-%m-%d')(兼容旧版本,但无法走索引)
— PHP 端用 date('Y-m-d', strtotime($row['created_at'])) 做二次归并也行,但不建议:增加 PHP 计算负担,且丢失数据库聚合能力
周和月分组要小心 ISO 周 vs 自定义周
统计“本周”或“上周”,别直接用 WEEK()——它默认按周日为起点,且受 mode 参数影响大:
— 要周一为每周开始,用 WEEK(created_at, 1)
— 更稳妥的是用 YEARWEEK(created_at, 1)(返回类似 202432),避免跨年周混乱
— 按月分组统一用 DATE_FORMAT(created_at, '%Y-%m'),比 YEAR(created_at)*100 + MONTH(created_at) 更直观、可读性强
查出来的数据空日期怎么补全?别在 PHP 循环里硬插
数据库只返回有数据的日期,图表横轴却需要连续时间线(比如 7 天都得有值,哪怕 count=0)。常见错误是在 PHP 里 for 循环 + in_array 判断补零,效率低还易错:
— 推荐方案:用 MySQL 生成日期序列(8.0+ 可用 CTE + RECURSIVE;5.7 用 JOIN 虚拟数字表)
— 更轻量做法:PHP 预生成日期数组(如 array_map(fn($i) => date('Y-m-d', strtotime("2024-01-01 + $i day")), range(0,6))),再用 array_merge_recursive() 或遍历合并查询结果
— 关键点:补零逻辑必须放在数据组装阶段,不能丢给前端 JS 处理——否则异步加载时缺日期会导致图表错位
曲线图数据格式要对齐前端库要求
多数 JS 图表库(Chart.js、ECharts)要的是两个平行数组:labels(时间字符串)和 data(数值),顺序严格对应:
— 不要用关联数组 ['2024-01-01' => 12, '2024-01-02' => 8] 直接输出,PHP json_encode() 后前端取值会乱序
— 正确做法:先按时间升序整理好键值对,再用 array_values() 提取 labels 和 data 两数组
— 注意时区:PHP date() 和 MySQL NOW() 时区不一致时,同一天可能被切到不同组,务必统一设为 date_default_timezone_set('Asia/Shanghai') 并确认 MySQL 的 time_zone 设置
立即学习“PHP免费学习笔记(深入)”;
时间分组本身不难,真正卡住人的永远是“空日期补全”和“时区对齐”——这两个点漏掉一个,图表就看起来像少了半截。










