Go 的 time.Parse 失败主因是 layout 格式错误,因其仅认参考时间“Mon Jan 2 15:04:05 MST 2006”对应格式,如“2006-01-02 15:04:05”正确,而“YYYY-MM-DD HH:mm:ss”非法;还需注意空白/BOM、时区匹配及优先使用 RFC3339 等预定义常量。
Go 的 time.Parse 不接受常见的 "YYYY-MM-DD HH:mm:ss" 这类占位符,它只认 Go 自己定义的参考时间:Mon Jan 2 15:04:05 MST 2006(即 Unix 时间戳 1136239445 的文字表示)。任何 layout 字符串都必须严格对应这个参考时间中各字段的位置和大小写。
"2006-01-02 15:04:05" ✅ 正确 —— 年份、月份、日期、小时(24 小时制)、分钟、秒都按参考时间对齐"YYYY-MM-DD HH:mm:ss" ❌ 错误 —— 这是其他语言(如 Python、JS)的习惯,Go 会直接返回 parsing time ... as "YYYY-MM-DD ...": cannot parse "Y" as "2"
"2006/01/02 3:04:05 PM" ✅ 可用 —— 对应参考时间中的 Jan 2 15:04:05,注意 3 是小时(12 小时制),后面必须带 PM 或 AM
01(带前导零)而非 1;星期用 Mon 而非 Monday;时区缩写用 MST(不是 UTC 或 GMT)——这些细节错一个就解析失败比如从 HTTP header、JSON 字段、数据库 TEXT 字段读出的时间字符串,可能开头/结尾有空白、BOM(\uFEFF)、全角空格(\u3000)等。而 time.Parse 遇到 layout 无法匹配的字符就立即报错,不会跳过或 trim。
strings.TrimSpace 清理首尾空白bytes.TrimPrefix([]byte(s), []byte("\uFEFF")) 剥离fmt.Printf("%q\n", []rune(s))" 2025-05-10T12:34:56Z " 直接传给 time.Parse("2006-01-02T15:04:05Z", s) 会失败;必须先 trimlayout 中没写时区标识(如 Z、MST、-0700),但输入字符串带时区,或者反过来,都会导致解析失败或结果偏差。
"2025-05-10T12:34:56+0800",layout 必须包含 "2006-01-02T15:04:05-0700"(注意 -0700 是固定格式,不能写成 +0800)"2025-05-10T12:34:56Z",layout 必须用 "2006-01-02T15:04:05Z",不能漏掉末尾的 Z
"2025-05-10 12:34:56"),但你希望按本地时区解释,需用 time.ParseInLocation 显式指定:loc, _ := time.LoadLocation("Asia/Shanghai")
t, err := time.ParseInLocation("2006-01-02 15:04:05", s, loc)Go 标准库提供了多个常用 layout 常量,比手写更安全,也更易读。尤其适合处理 HTTP、JSON、日志等标准格式时间。
time.RFC3339 → "2006-01-02T15:04:05Z07:00"(ISO 8601 子集,最常用)time.RFC3339Nano → 带纳秒精度的 RFC3339time.RFC1123 → "Mon, 02 Jan 2006 15:04:05 MST"(HTTP Date 头)time.Kitchen → "3:04PM"(仅时间,美式)CST),只支持缩写(MST)或偏移(-0700、Z);遇到 "CST"
真正麻烦的不是 layout 记不住,而是错误信息里不告诉你哪一位不匹配——它只报“cannot parse 'X' as 'Y'”,得靠比对参考时间逐字符检查。建议把常用 layout 写成常量集中管理,而不是每次临时拼字符串。