在开发过程中,我们有时会遇到这样的问题,将 2020-11-08T08:18:46+08:00 转成 2020-11-08 08:18:46,怎么解决这个问题?

解决这个问题,最好不要用字符串截取,或者说字符串截取是最笨的方法,这应该是时间格式化的问题。

我们先看一下 golang time 包中支持的 format 格式:

  1. const (
  2. ANSIC = "Mon Jan _2 15:04:05 2006"
  3. UnixDate = "Mon Jan _2 15:04:05 MST 2006"
  4. RubyDate = "Mon Jan 02 15:04:05 -0700 2006"
  5. RFC822 = "02 Jan 06 15:04 MST"
  6. RFC822Z = "02 Jan 06 15:04 -0700" // RFC822 with numeric zone
  7. RFC850 = "Monday, 02-Jan-06 15:04:05 MST"
  8. RFC1123 = "Mon, 02 Jan 2006 15:04:05 MST"
  9. RFC1123Z = "Mon, 02 Jan 2006 15:04:05 -0700" // RFC1123 with numeric zone
  10. RFC3339 = "2006-01-02T15:04:05Z07:00"
  11. RFC3339Nano = "2006-01-02T15:04:05.999999999Z07:00"
  12. Kitchen = "3:04PM"
  13. // Handy time stamps.
  14. Stamp = "Jan _2 15:04:05"
  15. StampMilli = "Jan _2 15:04:05.000"
  16. StampMicro = "Jan _2 15:04:05.000000"
  17. StampNano = "Jan _2 15:04:05.000000000"
  18. )

我们找到了 RFC3339 ,那就很简单了,我们封装一个方法 RFC3339ToCSTLayout,见下面代码。

  1. package timeutil
  2. import "time"
  3. var (
  4. cst *time.Location
  5. )
  6. // CSTLayout China Standard Time Layout
  7. const CSTLayout = "2006-01-02 15:04:05"
  8. func init() {
  9. var err error
  10. if cst, err = time.LoadLocation("Asia/Shanghai"); err != nil {
  11. panic(err)
  12. }
  13. }
  14. // RFC3339ToCSTLayout convert rfc3339 value to china standard time layout
  15. func RFC3339ToCSTLayout(value string) (string, error) {
  16. ts, err := time.Parse(time.RFC3339, value)
  17. if err != nil {
  18. return "", err
  19. }
  20. return ts.In(cst).Format(CSTLayout), nil
  21. }

运行一下

  1. RFC3339Str := "2020-11-08T08:18:46+08:00"
  2. cst, err := timeutil.RFC3339ToCSTLayout(RFC3339Str)
  3. if err != nil {
  4. fmt.Println(err)
  5. }
  6. fmt.Println(cst)

输出:

  1. 2020-11-08 08:18:46

小结

同理,若遇到 RFC3339NanoRFC822RFC1123 等格式,也可以使用类似的方法,只需要在 time.Parse() 中指定时间格式即可。