最近开发一个公司项目,发现Go语言结构体Json转换时,存在时间格式不一样问题。在网上找了很久也没有找到一个很好的方案。即结构体序列化后的格式是`1993-01-01T20:08:23.000000028+08:00`。但为了兼容公司以往的项目,希望沿用`1993-01-01 20:08:23`这种格式。网上找到了下面的代码,可以解决大部分的问题。
import "time" const ( DateFormat = "2006-01-02" TimeFormat = "2006-01-02 15:04:05" ) type Time time.Time func Now() Time { return Time(time.Now()) } func (t *Time) UnmarshalJSON(data []byte) (err error) { now,err := time.ParseInLocation(`"`+TimeFormat+`"`,string(data),time.Local) *t = Time(now) return } func (t Time) MarshalJSON() ([]byte,error) { b := make([]byte,len(TimeFormat)+2) b = append(b,'"') b = time.Time(t).AppendFormat(b,TimeFormat) b = append(b,'"') return b,nil } func (t Time) String() string { return time.Time(t).Format(TimeFormat) }
但是这样写会对原有的struct产生影响,需要将原来的time.Time的变量类型替换成Time。可在使用一些ORM时就不行了,比如Beego的Orm就会报错了。因此,要在不改变结构体时间类型的情况下,替换掉原来的时间格式。就只能和上面的代码一样,给结构体也实现MarshalJson和UnmarshalJson方法。
type User struct { Id int `json:"id"` Name string `json:"name"` CreatedAt time.Time `json:"created_at"` } func (u *User) MarshalJSON() ([]byte,error) { type Alias User user := &struct { CreatedAt Time `json:"created_at"` *Alias }{Time(u.CreatedAt),(*Alias)(u)} return json.Marshal(user) } func (u *User) UnmarshalJSON(data []byte) (err error) { type Alias User user := &struct { CreatedAt Time `json:"created_at"` *Alias }{Time(u.CreatedAt),(*Alias)(u)} err = json.Unmarshal(data,user) if err != nil { return err } user.Alias.CreatedAt = time.Time(user.CreatedAt) *u = User(*user.Alias) return nil } func main() { var user *User user = &User{ Id: 4,Name: "Liam",CreatedAt: time.Now(),} bytes,_ := json.Marshal(user) fmt.Printf("%v\n",string(bytes)) data := `{"id":3,"name":"Liam Lian","created_at":"2017-11-18 19:00:00"}` json.Unmarshal([]byte(data),&user) fmt.Printf("%v\n",user) }
虽然这样便实现了时间格式的兼容,而且不影响原来的结构体。但如果这样的结构体比较多的话,就会有很多的这类代码。目前还没找到完美的解决方案。