在Golang中解组错误格式化的日期时间

时间:2016-08-27 10:09:54

标签: json datetime go iso8601 rfc3339

背景

我正在学习Go,我正在尝试对日期时间进行一些JSON解组。

我有一些由我在C中编写的程序生成的JSON,我输出的是我认为有效的ISO8601 / RFC3339时区偏移。我正在使用strftime以下格式字符串:

%Y-%m-%dT%H:%M:%S.%f%z

(请注意strftime本身不支持%f,我有一个用纳秒替换它的包装器。)

然后会产生以下结果:

2016-08-08T21:35:14.052975+0200

然后在Go中解组这个不起作用: https://play.golang.org/p/vzOXbzAwdW

package main

import (
    "fmt"
    "time"
)

func main() {
    t, err := time.Parse(time.RFC3339Nano, "2016-08-08T21:35:14.052975+0200")
    if err != nil {
        panic(err)
    }
    fmt.Println(t)
}

输出:

panic: parsing time "2016-08-08T21:35:14.052975+0200" as "2006-01-02T15:04:05.999999999Z07:00": cannot parse "+0200" as "Z07:00"

(工作示例:https://play.golang.org/p/5xcM0aHsSw

这是因为RFC3339 expects the timezone offset to be in the format 02:00有一个:,但strftime将其输出为0200

所以我需要在我的C程序中修复它以输出正确的格式。

 %z     The +hhmm or -hhmm numeric timezone (that is, the hour and
              minute offset from UTC). (SU)

问题

但是,现在我有一堆JSON文件格式不正确:

2016-08-08T21:35:14.052975+0200

而不是正确的(在时区偏移中使用:):

2016-08-08T21:35:14.052975+02:00

但我仍然希望能够在我的Go程序中正确解组它。最好只有两个不同的JSON文件才能以完全相同的方式进行解析。

关于封送回JSON,应该使用正确的格式。

这就是我在struct中定义它的方式:

Time            time.Time `json:"time"`

所以问题是,这样做的“Go”方式是什么?

同样在我的代码示例中,我正在使用RFC3339Nano。我如何在结构的元数据中指定它?正如我现在只有json:"time"那样会忽略纳秒时间吗?

2 个答案:

答案 0 :(得分:14)

您可以定义支持这两种格式的自己的time字段类型:

type MyTime struct {
    time.Time
}

func (self *MyTime) UnmarshalJSON(b []byte) (err error) {
    s := string(b)

    // Get rid of the quotes "" around the value.
    // A second option would be to include them
    // in the date format string instead, like so below: 
    //   time.Parse(`"`+time.RFC3339Nano+`"`, s) 
    s = s[1:len(s)-1]

    t, err := time.Parse(time.RFC3339Nano, s)
    if err != nil {
        t, err = time.Parse("2006-01-02T15:04:05.999999999Z0700", s)
    }
    self.Time = t
    return
}

type Test struct {
    Time MyTime `json:"time"`
}

Try on Go Playground

在上面的示例中,我们采用预定义格式time.RFC3339Nano,其定义如下:

RFC3339Nano = "2006-01-02T15:04:05.999999999Z07:00"

并删除:

"2006-01-02T15:04:05.999999999Z0700"

time.Parse使用的时间格式如下所述: https://golang.org/pkg/time/#pkg-constants

另请参阅time.Parse的文档 https://golang.org/pkg/time/#Parse

P.S。在时间格式字符串中使用年份2006的事实可能是因为当年发布了Golang的第一个版本。

答案 1 :(得分:-2)

您可以尝试https://play.golang.org/p/IsUpuTKENg

package main

import (
    "fmt"
    "time"
)

func main() {
    t, err := time.Parse("2006-01-02T15:04:05.999999999Z0700", "2016-08-08T21:35:14.052975-0200")
    if err != nil {
        panic(err)
    }
    fmt.Println(t)
}