亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

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

 更新時間:2020年12月21日 09:37:35   作者:peihexian  
這篇文章主要介紹了golang xorm及time.Time自定義解決json日期格式的問題,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

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

type myTime time.Time ,然后針對myTime實現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
}

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

下面是我的對應數據庫表結構的struct 定義,

type ServerInfo struct {
 ServerInfoId       string   `xorm:"varchar(32) pk server_info_id"`
 CreatedAt        LocalTime `xorm:"timestamp created"`
 UpdatedAt        LocalTime `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是考慮到有時候數據值可能為NULL,即字段值可能為空的情況。

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

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

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

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日期格式處理一下數據庫里面的日期型數據解析!")
 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
}

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

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

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

除了需要修改xorm\statement.go文件的內容,還需要修改xorm\session_convert.go的第558行,增加以下代碼:

主要是增加下面的代碼

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

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

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

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

下面是加完以后的樣子

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

以上為個人經驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

相關文章

  • 詳解golang函數多返回值錯誤處理與error類型

    詳解golang函數多返回值錯誤處理與error類型

    這篇文章主要為大家詳細介紹了golang中函數多返回值錯誤處理與error類型的相關知識,文中的示例代碼簡潔易懂,感興趣的小伙伴快跟隨小編一起學習吧
    2023-10-10
  • 基于Golang實現內存數據庫的示例詳解

    基于Golang實現內存數據庫的示例詳解

    這篇文章主要為大家詳細介紹了如何基于Golang實現內存數據庫,文中的示例代碼講解詳細,具有一定的借鑒價值,需要的小伙伴可以參考一下
    2023-03-03
  • Golang設計模式工廠模式實戰(zhàn)寫法示例詳解

    Golang設計模式工廠模式實戰(zhàn)寫法示例詳解

    這篇文章主要為大家介紹了Golang 工廠模式實戰(zhàn)寫法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-08-08
  • Go語言變量創(chuàng)建的五種方法

    Go語言變量創(chuàng)建的五種方法

    這篇文章主要介紹了Go語言五種變量創(chuàng)建的方法,本文給大家提到了匿名變量的優(yōu)點,通過實例代碼給大家介紹的非常詳細,需要的朋友可以參考下
    2020-02-02
  • Golang中閉包與常用場景詳解

    Golang中閉包與常用場景詳解

    在很多的開源項目里,經??吹介]包的運用,這篇文章主要為大家簡要記錄一下閉包的概念和一些常用的場景,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-11-11
  • 用Go+Redis實現分布式鎖的示例代碼

    用Go+Redis實現分布式鎖的示例代碼

    在分布式的業(yè)務中 , 如果有的共享資源需要安全的被訪問和處理 , 那就需要分布式鎖,本文主要介紹了用Go+Redis實現分布式鎖的示例代碼,感興趣的可以了解一下
    2021-12-12
  • Go語言Elasticsearch數據清理工具思路詳解

    Go語言Elasticsearch數據清理工具思路詳解

    這篇文章主要介紹了Go語言Elasticsearch數據清理工具思路詳解,本文給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-10-10
  • go?map搬遷的實現

    go?map搬遷的實現

    本文主要介紹了go?map搬遷的實現,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2023-04-04
  • Go設計模式之備忘錄模式講解和代碼示例

    Go設計模式之備忘錄模式講解和代碼示例

    備忘錄是一種行為設計模式, 允許生成對象狀態(tài)的快照并在以后將其還原,本文就通過代碼示例給大家講講Go備忘錄模式,感興趣的小伙伴跟著小編一起來看看吧
    2023-08-08
  • Golang編譯器介紹

    Golang編譯器介紹

    今天小編就為大家分享一篇關于go語言編譯器的介紹,小編覺得內容挺不錯的,現在分享給大家,具有很好的參考價值,需要的朋友一起跟隨小編來看看吧
    2018-09-09

最新評論