DayJS 严格解析带时区的日期时间返回 false 的原因及解决方案

dayjs 在旧版本(如 1.1.17)中存在严格模式(strict parsing)下无法正确解析含时区 iso 格式字符串的 bug,升级至 1.1.19+ 后该问题已修复。

在从 Moment.js 迁移至 Day.js 的过程中,开发者常依赖 dayjs(date, formats, strict) 的严格解析能力来校验日期字符串是否完全匹配指定格式(尤其含时区的 ISO 8601 字符串)。但若你使用的是 Day.js ≤ v1.11.17,会发现如下代码始终返回 false:

const isValidTimestampDateTime = function (date) {
  const datetimeFormats = [
    "YYYY-MM-DDTHH:mm:ssZ",      // e.g., "2025-03-01T15:41:00+01:00"
    "YYYY-MM-DDTHH:mm:ss.SSSZ",  // e.g., "2025-03-22T11:51:53.766+01:00"
    "YYYY-MM-DDTHH:mm:ss[Z]",    // e.g., "2025-03-22T11:51:53Z"
  ];
  return dayjs(date, datetimeFormats, true).isValid(); // ❌ 在 v1.11.17 及更早版本中恒为 false
};

根本原因
Day.js v1.11.17 及之前版本中,strict 模式对带时区偏移量(如 +01:00、Z)的格式解析存在逻辑缺陷——解析器未能正确识别 Z 和 +HH:mm 等时区标记在多格式数组中的匹配行为,导致即使输入完全合法,isValid() 也返回 false。

解决方案
立即升级 Day.js 至 v1.11.19 或更高版本(推荐使用最新稳定版,如 v1.11.10+ 实际应为 v1.11.10 有误,正确最新版请以 official release 为准,当前(2025)主流稳定版为 v1.11.10 起已包含修复,但 v1.11.19 是明确修复该问题的里程碑版本):

# npm
npm install dayjs@^1.11.19

# yarn
yarn add dayjs@^1.11.19

升级后,上述代码即可正常工作:

console.log(isValidTimestampDateTime("2025-03-22T11:51:53.766+01:00")); // ✅ true
console.log(isValidTimestampDateTime("2025-03-22T11:51:53Z"));         // ✅ true
console.log(isValidTimestampDateTime("2025-03-22T11:51:53"));         // ❌ false(不匹配任一格式)

⚠️ 注意事项

  • Day.js 的多格式解析(dayjs(string, [format1, format2], strict))要求 输入字符串必须与某一格式完全精确匹配,包括字面量字符(如 T、Z)、分隔符和精度(.SSS 缺失则不匹配 SSSZ 格式);
  • Z 格式("YYYY-MM-DDTHH:mm:ss[Z]")仅匹配字面量 Z,不匹配 +00:00;如需兼容 +00:00 和 Z,建议统一使用 "YYYY-MM-DDTHH:mm:ss.SSSZ"(它同时支持 +01:00 和 Z);
  • 若项目需强校验 ISO 8601 兼容性,可额外结合 dayjs().toISOString() 输出验证,或使用 dayjs.parseZone() 处理带时区输入。

总结:此非用法错误,而是特定版本的已知缺陷。保持 Day.js 版本更新是保障严格解析可靠性的关键。