用 array_map 配合 date() 最稳妥,需显式设置时区、归一化输入(如 strtotime 处理字符串)、兜底非法值;避免循环内调用 date_default_timezone_set,数据库日期应统一转换后再格式化。

PHP批量把时间戳转成指定格式,用 array_map 最稳
直接用 array_map 配合 date 是最常见也最不容易出错的方式。它不修改原数组,返回新数组,语义清晰,也不依赖额外扩展。
- 别用
foreach手动循环再date()—— 容易漏掉键名处理或时区没统一 -
date('Y-m-d H:i:s', $timestamp)的第二个参数必须是整数,如果数组里混了字符串时间(如"2024-01-01"),会全变成1970-01-01 - 时区默认是系统设置的,线上和本地不一致时结果会差8小时,建议显式设
date_default_timezone_set('Asia/Shanghai')
date_default_timezone_set('Asia/Shanghai');
$timestamps = [1704067200, 1704153600, 1704240000];
$formatted = array_map(fn($t) => date('Y-m-d', $t), $timestamps);
// → ['2024-01-01', '2024-01-02', '2024-01-03']
数组里有各种时间格式(字符串、DateTime、时间戳),得先归一化
真实业务里,输入往往不干净:可能混着 "2024-01-01"、"2024-01-01 12:00:00"、DateTime 对象、甚至 null。这时候硬套 date() 会警告或失败。
- 优先用
strtotime()转成时间戳,它对多数常见字符串格式兼容性好;但注意它对0000-00-00或空字符串返回false,得兜底 -
DateTime::createFromFormat()更精确,适合固定格式(如只有Y/m/d),但写法啰嗦,且格式错一个字符就返回false - 遇到
null或非法值,别让整个数组崩掉,加一层判断返回空字符串或默认值
$mixed = ['2024-01-01', 1704067200, null, new DateTime('2024-01-03')];
$formatted = array_map(function($item) {
if ($item instanceof DateTime) {
return $item->format('Y-m-d');
}
$ts = is_numeric($item) ? (int)$item : strtotime($item);
return $ts === false ? '' : date('Y-m-d', $ts);
}, $mixed);
要格式化数据库查出来的日期字段,注意 MySQL 的 datetime 和 timestamp 差异
从 PDO 或 mysqli 拿到的结果里,MySQL 的 datetime 默认是字符串(如 "2024-01-01 10:30:00"),而 timestamp 可能被 PHP 自动转成时间戳(取决于PDO属性 PDO::ATTR_EMULATE_PREPARES 和 PDO::MYSQL_ATTR_USE_BUFFERED_QUERY)。
- 不要假设所有日期字段都是字符串 —— 开发环境开
emulate_prepared,生产关了,结果类型可能突变 - 稳妥做法:统一用
strtotime($row['created_at'])再格式化,它对字符串和整数都有效 - 如果用 Laravel/Eloquent,
$model->created_at->format('Y-m-d')更安全,因为 Eloquent 已帮你做了类型封装
性能敏感场景下,避免在循环里反复调用 date_default_timezone_set
这个函数不是无害的——它会重置整个进程的时区状态,频繁调用会影响其他并发请求的日期逻辑,尤其在常驻内存的 Swoole 或 PHP-FPM 子进程中。
立即学习“PHP免费学习笔记(深入)”;
- 时区只应在脚本开头设一次,不要放在循环或闭包里
- 如果不同数组要转不同时区,改用
DateTime对象:(new DateTime($ts))->setTimezone(new DateTimeZone('Asia/Tokyo'))->format('Y-m-d') -
date()比DateTime::format()快约 15%~20%,但差别只在十万级以上数据才明显;日常用哪个都行,别为这点性能牺牲可读性











