如何在 Go 中实现带序数后缀(如 1st、2nd、3rd)的英文日期格式化

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, 13

th),这是常见疏漏点——仅判断个位数会错误地将 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 包进行国际化增强;而简单英文序数场景,上述手动后缀拼接方式简洁、高效且完全可控。