strtotime() 解析失败主因是格式混乱或语义非法,非字符串长度;应优先用 DateTime::createFromFormat() 显式控制格式,并预处理混合文本提取标准日期。
strtotime() 解析日期只要字符串内容符合常见日期格式(如 "2025-03-15"、"15/03/2025"、"March 15, 2025"),哪怕前后有多余空格或少量无关字符,strtotime() 通常也能成功提取有效日期部分。它内部会跳过前导空白、尝试匹配已知模式,并在遇到无法识别的字符时截断——不是因为“太长”,而是因为“解析器提前放弃”。
比如这些看似“不长”但必然失败的字符串:
"2025-03-15T14:30:00+0800 extra text" —— 含 ISO 8601 时间但尾部杂字,strtotime() 在 PHP 8.0+ 中可能解析失败(旧版有时容忍)"15/13/2025" —— 日/月/年顺序中月为 13,格式合法但语义非法,返回 false
"2025-02-30" —— 二月没有 30 号,strtotime() 默认转成 2025-03-01(自动进位),不是报错"2025/03/15 14:30:xx" —— 秒字段含非数字,解析中断,返回 false
DateTime::createFromFormat() 替代 strtotime() 控制精度当输入字符串结构固定(如日志时间 "[2025-03-15 14:30:22] INFO..."),靠 strtotime() 猜容易翻车。应显式指定格式:
DateTime::createFromFormat('Y-m-d H:i:s', substr($str, 1, 19));
关键点:
substr() 或正则提取确定位置的子串,避免依赖自动识别DateTime::createFromFormat() 对格式严格,parse_error 可通过 DateTime::getLastErrors() 检查| 严格标记)strtotime() 更快,无全局时区隐式转换风险像 "订单创建于2025年03月15日,共3件商品" 这类字符串,strtotime() 基本无效。不能指望它从中文里“认出”年月日。
稳妥做法是:
preg_match('/(\d{4})[年\-\/](\d{1,2})[月\-\/](\d{1,2})[日\s]/u', $str, $m)
"{$m[1]}-{$m[2]}-{$m[3]}" 再传给 DateTime::createFromFormat()
strtotime(),尤其含中文、emoji、HTML 标签时,解析结果不可控
实际项目里,日期解析失败往往不是因为字符串“太长”,而是没人检查 $date === false 就往下用了。最常被忽略的,是没意识到 strtotime() 在遇到模糊输入(如 "01/02/03")时,会按服务器 locale 默认解释成美式或欧式,且不报错。