贝利信息

PHP空字符串转日期会报错吗_PHP防空串转日期异常【注意】

日期:2026-01-17 00:00 / 作者:星夢妙者
strtotime() 对空字符串返回 false 并可能触发警告,不静默转为默认日期;DateTime::createFromFormat() 更可控,需手动判断空输入;推荐封装 safeParseDate() 函数统一处理。

会报错,而且默认行为是返回 false 或触发警告(取决于时区设置和 PHP 版本),不是静默转成某个默认日期。

空字符串传给 strtotime() 的实际表现

PHP 的 strtotime() 对空字符串(''null、仅空白字符)没有明确定义的解析逻辑。它会尝试“猜”,但绝大多数情况下直接失败:

DateTime::createFromFormat() 处理空输入更可控

相比 strtotime()DateTime::createFromFormat() 要求显式格式,对空输入不“猜测”,更适合防御性编码:

if ($dateStr === '' || $dateStr === null || trim($dateStr) === '') {
    // 明确跳过或设为 null / 默认值
    $date = null;
} else {
    $date = DateTime::createFromFormat('Y-m-d', $dateStr);
    if (!$date || $date->format('Y-m-d') !== $dateStr) {
        // 格式不匹配或解析失败(例如 '2025-02-30')
        $date = null;
    }
}

关键点:createFromFormat() 不会把空串当作“今天”,也不会自动 fallback;必须手动判断输入有效性。

常见误用:直接用 new DateTime($str)

这个构造函数底层调用类似 strtotime() 的解析器,所以同样危险:

永远不要假设用户输入或数据库字段非空。哪怕字段定义为 NOT NULL,PHP 层收到的仍可能是空字符串(比如表单未填、JSON 解析后键存在但值为空)。

推荐做法:封装一个安全的日期解析函数

统一处理空、无效、模糊输入,避免散落在各处的 if (!empty())

function safeParseDate(string $input, string $format = 'Y-m-d'): ?DateTime
{
    $trimmed = trim($input);
    if ($trimmed === '') {
        return null;
    }

    $date = DateTime::createFromFormat($format, $trimmed);
    if (!$date || $date->format($format) !== $trimmed) {
        return null;
    }

    return $date;
}

// 使用
$dt = safeParseDate($_POST['birth_date'] ?? '');
if ($dt) {
    echo $dt->format('Y年m月d日');
} else {
    echo '日期格式错误或为空';
}

注意:这里用 ?DateTime 声明返回类型(PHP 7.1+),强制调用方处理 null 情况。很多线上 bug 就是因为忽略了 strtotime('') 返回 false,又没做 === false 判断,结果 false 被当成 0 当作时间戳去格式化,输出 1970 年。