我的数据模型定义了多个结构,它们都具有两个共同的字段:StartDate
和EndDate
。我需要将这两个字段在经过编组的JSON中格式化为2018-09-21
,因此结构实现了Marshaller
接口:
type Results struct {
Source string `json:"source"`
StartDate time.Time
EndDate time.Time
}
type WeightedResults struct {
Source string `json:"source"`
StartDate time.Time
EndDate time.Time
}
func (r Results) MarshalJSON() ([]byte, error) {
type Alias Results
if equalDate(r.StartDate, r.EndDate) {
return json.Marshal(&struct {
Date string `json:"date"`
Alias
}{
Date: r.StartDate.Format(dateFormat),
Alias: (Alias)(r),
})
}
return json.Marshal(&struct {
StartDate string `json:"start_date"`
EndDate string `json:"end_date"`
Alias
}{
StartDate: r.StartDate.Format("2006-01-02"),
EndDate: r.EndDate.Format("2006-01-02"),
Alias: (Alias)(r),
})
}
func (r WeightedResults) MarshalJSON() ([]byte, error) {
type Alias WeightedResults
if equalDate(r.StartDate, r.EndDate) {
return json.Marshal(&struct {
Date string `json:"date"`
Alias
}{
Date: r.StartDate.Format(dateFormat),
Alias: (Alias)(r),
})
}
return json.Marshal(&struct {
StartDate string `json:"start_date"`
EndDate string `json:"end_date"`
Alias
}{
StartDate: r.StartDate.Format("2006-01-02"),
EndDate: r.EndDate.Format("2006-01-02"),
Alias: (Alias)(r),
})
}
上面的解决方案可以正常工作,但是会产生很多代码重复。有什么方法可以重构MarshalJSON
的两个实现以使用相同的逻辑/代码?我很清楚Go还没有提供泛型,但是必须解决这个问题,对吗?
答案 0 :(得分:5)
您的自定义封送拆封器不应位于结构上,而应位于嵌入time.Time
的自定义类型上:
type MyTime struct {
time.Time
}
func (t MyTime) MarshalJSON() ([]byte, error) {
return json.Marshal(t.Format("2006-01-02"))
}
然后在所需的任何地方使用此类型。
type Results struct {
Source string `json:"source"`
StartDate MyTime
EndDate MyTime
}