strtotime() 遇无效日期串返回 false 而非 0 或 1970-01-01;须用 === false 显式判断,DateTime::createFromFormat() 需配合 getLastErrors() 和 round-trip 校验确保合法性。
strtotime() 是 PHP 最常用的字符串转时间戳函数,但它对非法日期串极其严格:比如 "2025-02-30"、"2025-13-01"、"abc" 这类输入,一律返回 false(不是 0),而非尝试“纠正”或降级处理。很多开发者误以为它会兜底,结果在条件判断中漏掉 false 检查,导致后续 date() 输出 1970 年日期。
false,不能只用 if ($ts)(因为 0 也会被判定为假)== 判断,应使用 === false,防止 0(1970-01-01 00:00:00)被误判相比 strtotime() 的模糊解析,DateTime::createFromFormat() 允许你指定格式并控制容错逻辑。关键点在于:默认不校验日期有效性——例如用 "Y-m-d" 解析 "2025-02-30",它会静默转成 2025-03-02(自动进位),这不是 bug,是设计行为。
DateTime::getLastErrors(),其中 error_count > 0 表示格式或逻辑错误$dt->format('Y-m-d') === $input 校验是否“原样可逆”(适合简单格式)createFromFormat() 使用默认时区,可能导致跨日偏差直接裸用系统函数容易踩坑,建议封装一层,统一处理常见无效情况(空、null、明显乱码、格式不符、逻辑非法)。以下是一个轻量实用版本:
function parseDate(string $input, string $format = 'Y-m-d'): ?DateTime
{
if (trim($input) === '') {
return null;
}
$dt = DateTime::createFromFormat($format, $input);
$errors = DateTime::getLastErrors();
if ($dt === false || $errors['error_count'] > 0 || $errors['warning_count'] > 0) {
return null;
}
// 额外校验:确保输出能 round-trip 回原始字符串(防 2025-02-30 → 2025-03-02)
if ($dt->format($format) !== $input) {
return null;
}
return $dt;
}
null 表示不可
信输入,调用方无需再判断真假值歧义"2025-01-01 " 尾部空格),避免隐式截断strtotime(),规避其过度“智能”的副作用(如把 "next Monday" 当合法输入)开发中常见场景:用户提交 "2025-02-30",PHP 用 strtotime() 转成 false,然后没处理就传给 MySQL 的 DATE 字段,结果 MySQL 可能插入 '0000-00-00' 或报错,取决于 SQL mode。
DateTime 实例存在且有效,再取 $dt->format('Y-m-d')
is_object($dt) && $dt instanceof DateTime 判断,比检查字符串更可靠DATE 类型DateTime::createFromFormat() 的警告(warning)不触发异常也不返回 false,但可能意味着输入含不可见字符、空格、或格式部分匹配——这些都得靠 getLastErrors() 主动捞出来。