time.Parse 解析失败时返回 *time.ParseError 类型的非 nil 错误,包含原始字符串、预期格式和出错位置等诊断信息,需显式检查 err != nil 而不可忽略或用 == 比较。
time.Parse 解析失败时不会 panic,而是返回一个非 nil 的 error 值,类型通常是 
*time.ParseError。这个错误包含原始字符串、预期格式、出错位置等信息,可用于诊断,但不能直接用 == 与预定义错误比较(它不是变量,是结构体指针)。
常见错误现象:
- 输入字符串为 "2025-13-01"(月份越界)→ 报 month out of range
- 格式写成 "2006/01/02" 但输入是 "2025-01-01" → 报 parsing time ... as ...: cannot parse ...
err != nil,不要忽略返回值errors.Is(err, xxx) 判断具体原因(*time.ParseError 没有导出的哨兵错误)err.Error(),生产环境建议记录原始输入 + 错误消息Go 标准库不提供“尝试多个格式”的内置函数,必须手动轮询。关键点是:避免重复解析成功字符串、控制尝试次数、明确优先级。
func parseTimeMulti(s string) (time.Time, error) {
formats := []string{
time.RFC3339,
"2006-01-02T15:04:05",
"2006-01-02 15:04:05",
"2006-01-02",
}
for _, format := range formats {
if t, err := time.Parse(format, s); err == nil {
return t, nil
}
}
return time.Time{}, fmt.Errorf("unable to parse time: %q", s)
}T 就跳过空格分隔格式)time.RFC3339 已包含时区,若输入无时区(如 "2025-01-01T12:00:00"),会失败;可补 "Z" 或改用 time.RFC3339Nano 等变体如果输入时间字符串不含时区(如 "2025-01-01 12:00:00"),time.Parse 默认按 time.UTC 解析,而非系统本地时区 —— 这常被误认为“解析错了”。要用 time.ParseInLocation 显式指定位置。
time.Local 获取本机时区:t, err := time.ParseInLocation(layout, s, time.Local)
time.Local:Docker 容器或跨服务器部署时,time.Local 可能是 UTC 或不可控时区RFC3339),或统一约定为 UTC 并文档化time.LoadLocation("Asia/Shanghai") 可能返回 error,需提前检查高频调用 time.Parse(如日志解析、API 批量入参)可能成为瓶颈,因为内部涉及字符串切片、数字转换、时区计算。标准库无缓存机制,每次都是全量解析。
strconv + time.Date 手动拆解(跳过格式校验和时区逻辑)github.com/araddon/dateparse 支持模糊匹配,但会增加依赖和不确定性Parse 引起(pprof profile)time.Parse 的错误路径开销远高于成功路径,异常输入多时延迟波动明显解析时间最麻烦的从来不是语法,而是隐含的时区假设和格式歧义。哪怕加了 err != nil 检查,也得想清楚:这个字符串本该带时区吗?它的“本地时间”到底指哪台机器的本地?