色综合图-色综合图片-色综合图片二区150p-色综合图区-玖玖国产精品视频-玖玖香蕉视频

您的位置:首頁技術(shù)文章
文章詳情頁

golang xorm及time.Time自定義解決json日期格式的問題

瀏覽:77日期:2024-05-30 11:49:53

golang默認(rèn)的time.Time類型在轉(zhuǎn)為json格式時不是常用的2019-05-08 10:00:01這種格式,解決辦法是自定義一個時間類型,例如

type myTime time.Time ,然后針對myTime實現(xiàn)Marshaler接口的MarshalJSON方法,例如:

package models import ( 'database/sql/driver' 'time') const localDateTimeFormat string = '2006-01-02 15:04:05' type LocalTime time.Time func (l LocalTime) MarshalJSON() ([]byte, error) { b := make([]byte, 0, len(localDateTimeFormat)+2) b = append(b, ’'’) b = time.Time(l).AppendFormat(b, localDateTimeFormat) b = append(b, ’'’) return b, nil} func (l *LocalTime) UnmarshalJSON(b []byte) error { now, err := time.ParseInLocation(`'`+localDateTimeFormat+`'`, string(b), time.Local) *l = LocalTime(now) return err}

上面的代碼在網(wǎng)上隨手一搜就能找到,沒有什么困難的,接下來的才是本篇文章的重點,這玩意結(jié)合xorm使用時,特別是字段類型為*LocalTime的時候才需要折騰一番。

下面是我的對應(yīng)數(shù)據(jù)庫表結(jié)構(gòu)的struct 定義,

type ServerInfo struct { ServerInfoId string `xorm:'varchar(32) pk server_info_id'` CreatedAtLocalTime `xorm:'timestamp created'` UpdatedAtLocalTime `xorm:'timestamp updated'` DeletedAt*LocalTime `xorm:'timestamp deleted index'` OrgId string `xorm:'varchar(100) org_id' json:'orgId'` ServerIp string `xorm:'varchar(128) server_ip' json:'serverIp'` ServerNameDesc string `xorm:'varchar(500) server_name_desc' json:'serverNameDesc'`ServerTimeNow LocalTime `xorm:'timestamp server_time' json:'serverTime'` DataReceiveTime LocalTime `xorm:'timestamp data_receive_time' sql:'DEFAULT:current_timestamp' json:'dataRecvTime'` LastUploadDataTime *LocalTime `xorm:'timestamp last_upload_data_time' json:'lastUploadDataTime'` LastCheckTime *LocalTime `xorm:'timestamp last_check_time' json:'lastCheckTime'` LastErrorTime *LocalTime `xorm:'timestamp last_error_time' json:'lastErrorTime'` }

注意上面的字段類型,既有LocalTime類型的,又有*LocalTime類型的,*LocalTime是考慮到有時候數(shù)據(jù)值可能為NULL,即字段值可能為空的情況。

xorm不知道如何為LocalTime這個自定義類型進(jìn)行賦值或者取值,因此需要實現(xiàn)xorm的core包中的Conversion接口,這個接口的定義如下:

golang xorm及time.Time自定義解決json日期格式的問題

注意,坑已經(jīng)隱藏在上面的接口定義中了,過一會說。

整個完整的自定義時間類型的代碼變成了下面的這樣:

package models import ( 'database/sql/driver' 'time') const localDateTimeFormat string = '2006-01-02 15:04:05' type LocalTime time.Time func (l LocalTime) MarshalJSON() ([]byte, error) { b := make([]byte, 0, len(localDateTimeFormat)+2) b = append(b, ’'’) b = time.Time(l).AppendFormat(b, localDateTimeFormat) b = append(b, ’'’) return b, nil} func (l *LocalTime) UnmarshalJSON(b []byte) error { now, err := time.ParseInLocation(`'`+localDateTimeFormat+`'`, string(b), time.Local) *l = LocalTime(now) return err} func (l LocalTime) String() string { return time.Time(l).Format(localDateTimeFormat)} func (l LocalTime)Now()(LocalTime){ return LocalTime(time.Now())} func (l LocalTime)ParseTime(t time.Time)(LocalTime){ return LocalTime(t)} func (j LocalTime) format() string { return time.Time(j).Format(localDateTimeFormat)} func (j LocalTime) MarshalText() ([]byte, error) { return []byte(j.format()), nil} func (l *LocalTime) FromDB(b []byte) error { if nil == b || len(b) == 0 { l = nil return nil } var now time.Time var err error now, err = time.ParseInLocation(localDateTimeFormat, string(b), time.Local) if nil == err { *l = LocalTime(now) return nil } now, err = time.ParseInLocation('2006-01-02T15:04:05Z', string(b), time.Local) if nil == err { *l = LocalTime(now) return nil } panic('自己定義個layout日期格式處理一下數(shù)據(jù)庫里面的日期型數(shù)據(jù)解析!') return err} //func (t *LocalTime) Scan(v interface{}) error {// // Should be more strictly to check this type.// vt, err := time.Parse('2006-01-02 15:04:05', string(v.([]byte)))// if err != nil {// return err// }// *t = LocalTime(vt)// return nil//} func (l *LocalTime) ToDB() ([]byte, error) { if nil == l { return nil,nil } return []byte(time.Time(*l).Format(localDateTimeFormat)), nil} func (l *LocalTime) Value() (driver.Value, error) { if nil==l { return nil, nil } return time.Time(*l).Format(localDateTimeFormat), nil}

此時,要是數(shù)據(jù)庫的字段內(nèi)容都有值的話插入和更新應(yīng)該是沒有什么問題,但是*LocalTime字段的值為nil的話問題就開始出現(xiàn)了,上面說了,ToDB()方法的返回值類型為[]byte,當(dāng)字段值為nil時,返回nil看上去一切正常,但是xorm打印出來的sql語句數(shù)據(jù)值是下面這個樣子的:

golang xorm及time.Time自定義解決json日期格式的問題

這個[]uint8(nil)就是*LocalTime值為nil時的情況,數(shù)據(jù)庫驅(qū)動是不認(rèn)可[]uint8(nil)這種數(shù)據(jù)去寫給timestamp類型字段的,問題的根源就是ToDB方法的返回值類型為[]byte,既然是這樣,就需要我們?nèi)藶榈陌裑]uint8(nil)這種類型改為interface(nil)類型,數(shù)據(jù)庫驅(qū)動會識別interface(nil)為NULL值,修改代碼xormstatement.go第322行,把原來的val=data改成下面的樣子:

golang xorm及time.Time自定義解決json日期格式的問題

就是把val=data改為 if nil==data { val=nil } else {val=data} ,看上去邏輯沒有什么變化,但是給val=nil賦值的時候,val的類型就從[]uint8(nil)變成了interface(nil)了,這樣數(shù)據(jù)庫驅(qū)動就可以正確處理空值了。

除了需要修改xormstatement.go文件的內(nèi)容,還需要修改xormsession_convert.go的第558行,增加以下代碼:

golang xorm及time.Time自定義解決json日期格式的問題

主要是增加下面的代碼

//fix when pointer type value is null,added by peihexian,2019-05-07if nil==data { return nil,nil}

之所以加這個代碼是因為xorm作者沒有考慮指針類型字段值為nil的情況,xorm對有轉(zhuǎn)換的字段要么當(dāng)成數(shù)字,要么當(dāng)成了字符串,這兩種對于NULL類型的值都不適用,所以需要增加if nil==data return nil,nil這樣的代碼,還是把數(shù)據(jù)值組織成interface(nil)去給數(shù)據(jù)庫驅(qū)動去處理。

另外還有一個地方,是session_convert.go 第556行,同樣需要增加

if nil==data { //edit by peihexian 2019.06.19 return nil,nil}

下面是加完以后的樣子

golang xorm及time.Time自定義解決json日期格式的問題

到這里,對xorm做了幾處小的修改,自定義日期的問題及json格式化問題完美解決。

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持好吧啦網(wǎng)。如有錯誤或未考慮完全的地方,望不吝賜教。

標(biāo)簽: JavaScript
相關(guān)文章:
主站蜘蛛池模板: 欧美特黄一级视频 | 亚洲综合色一区二区三区另类 | 亚洲第一欧美 | 欧美日韩视频一区二区三区 | 日本午夜小视频 | 日本三级毛片 | 99re在线视频精品 | 亚洲国产高清在线 | 26uuu天天夜夜综合 | 免费看成人毛片日本久久 | 精品视频h | 亚洲在线中文 | 国产精品_国产精品_国产精品 | 千涩成人网 | 久久99久久精品免费思思 | 国产看片视频 | 亚洲美女综合网 | 午夜精品在线 | 黄色三级免费 | 91大神在线精品视频一区 | 国产亚洲网站 | 日本天堂在线视频 | 成人性色生活片全黄 | 俄罗斯aa毛片一级 | 超薄肉色丝袜精品足j福利 超级乱淫视频aⅴ播放视频 | 波多野结衣在线视频免费观看 | 亚洲高清中文字幕一区二区三区 | 午夜爽爽性刺激一区二区视频 | 国内精品免费一区二区三区 | 久久国产精品-国产精品 | 亚洲日本欧美在线 | 亚洲成成品网站有线 | 日韩亚洲欧美一区二区三区 | 九九这里只精品视在线99 | 亚洲国产成人私人影院 | 午夜限制r级噜噜片一区二区 | 手机在线看a | 国产高清久久 | 九色porny真实丨国产18 | 国产精品国色综合久久 | 中文在线三级中文字幕 |