关键是要将日期统一转为时间戳再比较,避免字符串比较出错;优先用DateTime::createFromFormat()显式解析不同格式,失败时需检查;最后用array_filter()配合时间戳范围筛选。

PHP 数组里筛选日期范围,关键不是用什么“高级函数”,而是把日期统一转成可比较的时间戳或标准格式——否则字符串比较会出错(比如 "2023-10-2" > "2023-10-15" 居然为真)。
确保数组中日期字段是标准格式或可转换
如果原始数组的日期是字符串(如 "2023-09-15"、"15/10/2023" 或 "Oct 12, 2023"),必须先统一解析。不能直接用 strtotime() 硬转所有格式——它对模糊格式(如 "01/02/2023")可能误判为美式/欧式。
- 优先使用
DateTime::createFromFormat()显式指定格式,例如DateTime::createFromFormat('d/m/Y', $dateStr) - 若格式不固定,先用正则粗筛或加上下文判断,再分情况解析
- 解析失败时
DateTime返回false,务必检查,避免后续getTimestamp()报错
用 array_filter() + 时间戳比较做高效筛选
时间戳(int)比较比字符串快且无歧义。把起止日期也转成时间戳,再在回调里比大小——这是最稳也最常见的做法。
$start = strtotime('2023-09-01');
$end = strtotime('2023-09-30');
$filtered = array_filter($data, function ($item) use ($start, $end) {
$ts = strtotime($item['date'] ?? '');
return $ts !== false && $ts >= $start && $ts <= $end;
});
-
strtotime()对Y-m-d格式兼容好,但对非标准格式(如带中文、多余空格)会返回false -
use ($start, $end)是必须的,闭包无法自动访问外部变量 - 注意
array_filter()保留原键名,如需重索引,后面加array_values()
处理时区和日期边界问题
本地服务器时区和业务时区不一致时,strtotime('2023-09-01') 可能生成当天 00:00:00 的 UTC 时间戳,导致筛选偏移一天。
立即学习“PHP免费学习笔记(深入)”;
- 显式设置时区:在解析前调用
date_default_timezone_set('Asia/Shanghai') - 起始时间建议用
strtotime('2023-09-01 00:00:00'),结束时间用strtotime('2023-09-30 23:59:59'),避免漏掉当天最后一秒数据 - 如果数据本身带时区信息(如
"2023-09-01T14:30:00+08:00"),用new DateTime($item['date'])更可靠
真正容易卡住的不是语法,而是日期字符串格式混乱 + 时区没对齐 + 没检查解析失败。先跑通一个样本数据的 var_dump(DateTime::createFromFormat(...)),再套进 array_filter,比硬写逻辑安全得多。











