
go 的 time.format() 不支持自动添加日期序数后缀(如 "1st"、"2nd"),需手动拼接后缀逻辑,再结合标准布局字符串完成格式化。
Go 标准库的 time.Time.Format() 方法基于固定布局(layout)字符串工作,其底层使用的是“参考时间” Mon Jan 2 15:04:05 MST 2006 的占位符映射机制。它不支持动态修饰符(如英语中 1st/2nd/3rd/4th 这类依赖数值规则的序数后缀),因此直接使用 "Monday 2nd January" 会导致 2nd 被当作字面量处理——其中 2 被解析为日字段(对应 t.Day()),而 nd 则原样输出,造成 4nd、1nd 等错误结果。
要正确实现“Wednesday 4th March”或“Sunday 1st March”,需将日期格式拆解为两部分:
- 使用 t.Format() 渲染静态部分(如 "Monday " 和 " January");
- 单独提取 t.Day(),根据英语序数规则计算后缀(st/nd/rd/th),再拼接进字符串。
以下是推荐的健壮实现:
func formatVerboseDate(t time.Time) string {
day := t.Day()
var suffix string
switch {
case day%100 >= 11 && day%100 <= 13:
suffix = "th" // 11th, 12th, 13th override all others
case day%10 == 1:
suffix = "st"
case day%10 == 2:
suffix = "nd"
case day%10 == 3:
suffix = "rd"
default:
suffix = "th"
}
return t.Format("Monday ") + fmt.Sprintf("%d%s", day, suffix) + t.Format(" January 2006")
}✅ 注意:此实现正确处理了 11–13 的特殊规则(如 11th, 12th, 13th),这是常见疏漏点——仅判断个位数会错误地将 11 判为 11st。
使用示例:
t1 := time.Date(2015, time.March, 4, 0, 0, 0, 0, time.UTC) t2 := time.Date(2015, time.March, 1, 0, 0, 0, 0, time.UTC) t3 := time.Date(2015, time.March, 11, 0, 0, 0, 0, time.UTC) fmt.Println(formatVerboseDate(t1)) // Wednesday 4th March 2015 fmt.Println(formatVerboseDate(t2)) // Sunday 1st March 2015 fmt.Println(formatVerboseDate(t3)) // Tuesday 11th March 2015
? 小结:Go 的时间格式化强调确定性与无状态,因此不内置语言敏感的智能格式。对本地化或复杂格式需求(如带序数、多语言月份/星期),建议结合 golang.org/x/text 包进行国际化增强;而简单英文序数场景,上述手动后缀拼接方式简洁、高效且完全可控。










