GO項(xiàng)目實(shí)戰(zhàn)之Gorm格式化時(shí)間字段實(shí)現(xiàn)
goshop開源項(xiàng)目的更新
備注:前面項(xiàng)目中用到的代碼已經(jīng)分享到GitHub中去了,并且以后所有項(xiàng)目中會(huì)出現(xiàn)的代碼都會(huì)提交上去,歡迎查閱。感興趣的可以點(diǎn)個(gè)star哦~
https://gitee.com/jobhandsome/goshop/
在使用 gorm
查詢時(shí),如果未對(duì)時(shí)間字段進(jìn)行處理,結(jié)構(gòu)體內(nèi)的字段類型咱們使用的是 time.Time
:
type Model struct { ID int64 `json:"id" gorm:"primary_key"` CreatedAt *time.Time `json:"created_at"` UpdatedAt *time.Time `json:"updated_at"` DeletedAt *time.Time `json:"deleted_at" sql:"index"` }
這里咱們使用 time.Time
類型在 gorm
進(jìn)行查詢的返回結(jié)果,讀取到的時(shí)間字段往往是這樣:“2022-07-03T22:14:02.973528+08:00
”,帶著時(shí)區(qū)和毫秒。但其實(shí)往往這樣的格式,不是咱們想要的。
那么問(wèn)題就來(lái)了:
- 如果想要 “
2022-07-03 22:14:02
” 這樣的格式,需要怎么處理呢? - 當(dāng)插入一條數(shù)據(jù)到對(duì)應(yīng)的表中時(shí),
UpdateAt
字段是不賦值的,插入到數(shù)據(jù)庫(kù)則會(huì)0001-01-01 00:00:00.000000+00:00
,系統(tǒng)賦了?個(gè)默認(rèn)值,當(dāng)不想插?默認(rèn)值
時(shí)如何處理?
通過(guò)上面的分析,咱們能確定兩個(gè)需求:
讀取到的時(shí)間需要是:“2022-07-03 22:14:02” 這樣的格式
當(dāng)時(shí)間字段不賦值時(shí),不插入默認(rèn)值
解決方法:
定義一個(gè)時(shí)間類型 struct
type LocalTime time.Time
雖然該數(shù)據(jù)類型實(shí)際類型為 time.Time
,但是不具備 time.Time
的內(nèi)置?法,需要重寫 MarshalJSON
?法來(lái)實(shí)現(xiàn)數(shù)據(jù)解析
func (t *LocalTime) MarshalJSON() ([]byte, error) { tTime := time.Time(*t) return []byte(fmt.Sprintf("\"%v\"", tTime.Format("2006-01-02 15:04:05"))), nil }
注意:
GO
的格式化時(shí)間的規(guī)定時(shí)間字符串必須為2006-01-02 15:04:05
這是GO
的誕?時(shí)間,不能更改為其他時(shí)間(這個(gè)時(shí)間字符串與java
的"yyyy-MM-dd HH:mm:ss
")同作?
將 time.Time
替換成 LocalTime
type Model struct { ID int64 `json:"id" gorm:"primary_key"` CreatedAt *LocalTime `json:"created_at"` UpdatedAt *localTime `json:"updated_at"` DeletedAt *localTime `json:"deleted_at" sql:"index"` }
到了這一步就解決了第一個(gè)需求讀取數(shù)據(jù)時(shí)將將時(shí)間數(shù)據(jù)格式化。
下面來(lái)實(shí)現(xiàn)第二個(gè)需求:
func (t LocalTime) Value() (driver.Value, error) { var zeroTime time.Time tlt := time.Time(t) //判斷給定時(shí)間是否和默認(rèn)零時(shí)間的時(shí)間戳相同 if tlt.UnixNano() == zeroTime.UnixNano() { return nil, nil } return tlt, nil }
Value
?法即在存儲(chǔ)時(shí)調(diào)?,將該?法的返回值進(jìn)?存儲(chǔ),該?法可以實(shí)現(xiàn)數(shù)據(jù)存儲(chǔ)前對(duì)數(shù)據(jù)進(jìn)?相關(guān)操作。
func (t *LocalTime) Scan(v interface{}) error { if value, ok := v.(time.Time); ok { *t = LocalTime(value) return nil } return fmt.Errorf("can not convert %v to timestamp", v) }
Scan
?法可以實(shí)現(xiàn)在數(shù)據(jù)查詢出來(lái)之前對(duì)數(shù)據(jù)進(jìn)?相關(guān)操作。
到了這一步,咱們就實(shí)現(xiàn)了上面需求的功能。
補(bǔ)充:gorm時(shí)間格式化問(wèn)題詳解
說(shuō)明
在做項(xiàng)目時(shí)發(fā)現(xiàn)gorm的時(shí)間格式是帶有時(shí)區(qū)輸入輸出的,對(duì)平常使用的2020-01-03 12:22:33格式有一定的出入,不方便前端和后端的對(duì)接,所以自己整理一下處理這個(gè)問(wèn)題方法,方便大家參考
代碼如下
package models import ( "database/sql/driver" "errors" "fmt" "strings" "time" ) //BaseModel 基礎(chǔ)結(jié)構(gòu)體 信息信息 type BaseModel struct { CreateTime MyTime `gorm:"comment:'創(chuàng)建時(shí)間';type:timestamp;";json:"createTime"` UpdateTime MyTime `gorm:"comment:'修改時(shí)間';type:timestamp;";json:"updateTime"` Remark string `gorm:"comment:'備注'";json:"remark"` } //MyTime 自定義時(shí)間 type MyTime time.Time func (t *MyTime) UnmarshalJSON(data []byte) error { if string(data) == "null" { return nil } var err error //前端接收的時(shí)間字符串 str := string(data) //去除接收的str收尾多余的" timeStr := strings.Trim(str, "\"") t1, err := time.Parse("2006-01-02 15:04:05", timeStr) *t = MyTime(t1) return err } func (t MyTime) MarshalJSON() ([]byte, error) { formatted := fmt.Sprintf("\"%v\"", time.Time(t).Format("2006-01-02 15:04:05")) return []byte(formatted), nil } func (t MyTime) Value() (driver.Value, error) { // MyTime 轉(zhuǎn)換成 time.Time 類型 tTime := time.Time(t) return tTime.Format("2006-01-02 15:04:05"), nil } func (t *MyTime) Scan(v interface{}) error { switch vt := v.(type) { case time.Time: // 字符串轉(zhuǎn)成 time.Time 類型 *t = MyTime(vt) default: return errors.New("類型處理錯(cuò)誤") } return nil } func (t *MyTime) String() string { return fmt.Sprintf("hhh:%s", time.Time(*t).String()) }
實(shí)現(xiàn)原理
其實(shí)現(xiàn)方式其實(shí)是通過(guò)在時(shí)間讀取json時(shí)對(duì)解析格式進(jìn)行重寫,對(duì)轉(zhuǎn)換成字符串時(shí)進(jìn)行重寫達(dá)到效果
總結(jié)
到此這篇關(guān)于GO項(xiàng)目實(shí)戰(zhàn)之Gorm格式化時(shí)間字段實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)GO Gorm格式化時(shí)間字段內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
- 如何使用Go語(yǔ)言獲取當(dāng)天、昨天、明天、某天0點(diǎn)時(shí)間戳以及格式化時(shí)間
- golang gorm中格式化時(shí)間問(wèn)題詳解
- go語(yǔ)言中時(shí)間戳格式化的方法
- golang 使用time包獲取時(shí)間戳與日期格式化操作
- golang中time包之時(shí)間間隔格式化和秒、毫秒、納秒等時(shí)間戳格式輸出的方法實(shí)例
- 詳解Go如何優(yōu)雅的對(duì)時(shí)間進(jìn)行格式化
- django rest framework serializer返回時(shí)間自動(dòng)格式化方法
- go語(yǔ)言中g(shù)orm時(shí)間格式化
- golang中的時(shí)間格式化
- 提升編程技能:學(xué)習(xí)如何在Go語(yǔ)言中正確格式化時(shí)間
相關(guān)文章
golang框架中跨服務(wù)的最佳通信協(xié)議和工具
在 go 框架中實(shí)現(xiàn)跨服務(wù)通信的最佳實(shí)踐包括使用 grpc(適用于低延遲高吞吐量)、http 客戶端(適用于 restful api)和消息隊(duì)列(適用于異步解耦通信),在選擇通信方式時(shí),應(yīng)考慮服務(wù)交互模式、性能要求和部署環(huán)境等因素2024-06-06go語(yǔ)言實(shí)現(xiàn)Elasticsearches批量修改查詢及發(fā)送MQ操作示例
這篇文章主要為大家介紹了go語(yǔ)言實(shí)現(xiàn)Elasticsearches批量修改查詢及發(fā)送MQ操作示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-04-04golang 64位linux環(huán)境下編譯出32位程序操作
這篇文章主要介紹了golang 64位linux環(huán)境下編譯出32位程序操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12使用Go語(yǔ)言解決Scan空格結(jié)束輸入問(wèn)題
這篇文章主要為大家介紹了使用Go語(yǔ)言來(lái)解決Scan空格結(jié)束輸入問(wèn)題,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步2021-11-11golang?熔斷器的實(shí)現(xiàn)過(guò)程
這篇文章主要介紹了golang?熔斷器的實(shí)現(xiàn)過(guò)程,Go?項(xiàng)目中使用熔斷技術(shù)提高系統(tǒng)容錯(cuò)性。接下倆就來(lái)給打家介紹?go?熔斷器和其使用,需要的朋友可以參考一下2022-01-01Golang使用Apache PLC4X連接modbus的示例代碼
Modbus是一種串行通信協(xié)議,是Modicon公司于1979年為使用可編程邏輯控制器(PLC)通信而發(fā)表,這篇文章主要介紹了Golang使用Apache PLC4X連接modbus的示例代碼,需要的朋友可以參考下2024-07-07