在golang xorm中使用postgresql的json,array類型的操作
xorm支持各種關(guān)系數(shù)據(jù)庫(kù),最近使用postgresql時(shí),總是踩到一些坑,在此記錄下解決方式。
在使用postgresql的array類型時(shí),查詢有點(diǎn)問(wèn)題,xorm的官方文檔給出重寫(xiě)的方式,但是不是很清晰:
官方文檔鏈接:http://xorm.io/docs
也就是說(shuō)碰到基礎(chǔ)庫(kù)不支持的類型,需要我們?nèi)ブ貙?xiě)ToDB、FromDB方法,廢話不多說(shuō)直接上代碼:
比如int8[]類型,自定一個(gè)Int64Array
type Int64Array []int64 func (s *Int64Array) FromDB(bts []byte) error { if len(bts) == 0 { return nil } str := string(bts) if strings.HasPrefix(str, "{") { str = "[" + str[1:len(str)] } if strings.HasSuffix(str, "}") { str = str[0: len(str)-1] + "]" } var ia = &[]int64{} err := json.Unmarshal([]byte(str), ia) if err != nil { return err } *s = Int64Array(*ia) return nil } func (s *Int64Array) ToDB() ([]byte, error) { return serializeBigIntArray(*s, "{", "}"), nil } func (arr Int64Array) MarshalJSON() ([]byte, error) { return serializeBigIntArrayAsString(arr, "[", "]"), nil } func (arr *Int64Array) UnmarshalJSON(b []byte) error { var strarr []string var intarr []int64 err := json.Unmarshal(b, &strarr) if err != nil { return err } for _, s := range strarr { i, err := strconv.ParseInt(s, 10, 64) if err != nil { return err } intarr = append(intarr, i) } *arr = intarr return nil } func serializeBigIntArray(s []int64, prefix string, suffix string) []byte { var buffer bytes.Buffer buffer.WriteString(prefix) for idx, val := range s { if idx > 0 { buffer.WriteString(",") } buffer.WriteString(strconv.FormatInt(val, 10)) } buffer.WriteString(suffix) return buffer.Bytes() } func serializeBigIntArrayAsString(s []int64, prefix string, suffix string) []byte { var buffer bytes.Buffer buffer.WriteString(prefix) for idx, val := range s { if idx > 0 { buffer.WriteString(",") } buffer.WriteString("\"") buffer.WriteString(strconv.FormatInt(val, 10)) buffer.WriteString("\"") } buffer.WriteString(suffix) return buffer.Bytes() }
json類型:
type Cover struct { Id int64 `json:"id,omitempty"` Fid string `json:"fid,omitempty"` Type int8 `json:"type,omitempty"` Url string `json:"url,omitempty"` } func (c *Cover) FromDB(bytes []byte) error { return json.Unmarshal(bytes, c) } func (c *Cover) ToDB() (bytes []byte, err error) { bytes, err = json.Marshal(c) return }
具體使用:
type Course struct { Id int64 `json:"id,string" form:"id"` Name string `json:"name" form:"name"` Brief string `json:"brief" form:"brief"` Description string `json:"description" form:"description"` Cover common.Cover `xorm:"Text" json:"cover" form:"cover"` Categories common.Int64Array `xorm:"Text" json:"categories" form:"categories[]"` Tags common.Int64Array `json:"tags" form:"tags[]"` Difficulty float64 `json:"difficulty" form:"difficulty"` Price float64 `json:"price" form:"price"` Markets common.Int64Array `json:"markets" form:"markets[]"` StudentAmount int64 `json:"studentAmount" form:"studentAmount"` SubjectAmount int64 `json:"subjectAmount" form:"subjectAmount"` Crt time.Time `json:"crt"` Lut time.Time `json:"lut"` Status int16 `json:"status" form:"status"` common.Page `xorm:"-"` }
補(bǔ)充:golang gin xorm注意事項(xiàng)
1. 無(wú)論是golang還是xorm中,在填寫(xiě)j'son字段時(shí),注意空格,比如 `json:"abcd "` `json:"abcd"`是不一樣的,不仔細(xì)對(duì)比會(huì)出錯(cuò)
2.當(dāng)結(jié)合gin框中的
c.JSON(http.StatusOK,gin.H{})操作
并且使用xorm中的join,find操作時(shí)(https://www.kancloud.cn/xormplus/xorm/167102)要注意如下現(xiàn)象,
假如定義兩個(gè)結(jié)構(gòu)體對(duì)應(yīng)兩個(gè)表
然后使用聯(lián)合查詢,先把兩個(gè)結(jié)構(gòu)體結(jié)合成一個(gè)結(jié)構(gòu)體,假如如下,在UserGroup中使用User和Group匿名結(jié)構(gòu)體,
那么當(dāng)我們使用gin的c.JSON(http.StatusOK,gin.H{"data":UserGroup})返回?cái)?shù)據(jù)時(shí)會(huì)導(dǎo)致Group和User中同名字段顯示不了,這應(yīng)該是gin和xorm的不是很兼容造成的(沒(méi)有深究),
要解決這個(gè)問(wèn)題,最好讓UserGroup中的User和Group不要以匿名結(jié)構(gòu)體的形式存在
可以改成
type UserGroup struct { MyUser User `xorm:"extends" json:"你要json中返回的名字"` MyGroup Group `xorm:"extends" json:"你要json中返回的名字"` }
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
相關(guān)文章
Golang設(shè)計(jì)模式工廠模式實(shí)戰(zhàn)寫(xiě)法示例詳解
這篇文章主要為大家介紹了Golang 工廠模式實(shí)戰(zhàn)寫(xiě)法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-08-08Go語(yǔ)言LeetCode題解706設(shè)計(jì)哈希映射
這篇文章主要為大家介紹了Go語(yǔ)言LeetCode題解706設(shè)計(jì)哈希映射示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12Goland使用delve進(jìn)行遠(yuǎn)程調(diào)試的詳細(xì)教程
網(wǎng)上給出的使用delve進(jìn)行遠(yuǎn)程調(diào)試,都需要先在本地交叉編譯或者在遠(yuǎn)程主機(jī)上編譯出可運(yùn)行的程序,然后再用delve在遠(yuǎn)程啟動(dòng)程序,本教程會(huì)將上面的步驟簡(jiǎn)化為只需要兩步,1,在遠(yuǎn)程運(yùn)行程序2,在本地啟動(dòng)調(diào)試,需要的朋友可以參考下2024-08-08Go語(yǔ)言利用Unmarshal解析json字符串的實(shí)現(xiàn)
本文主要介紹了Go語(yǔ)言利用Unmarshal解析json字符串的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-05-05解決go mod私有倉(cāng)庫(kù)拉取的問(wèn)題
這篇文章主要介紹了解決go mod私有倉(cāng)庫(kù)拉取的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-05-05一文教你如何快速學(xué)會(huì)Go的切片和數(shù)組數(shù)據(jù)類型
數(shù)組是屬于同一類型的元素的集合。切片是數(shù)組頂部的方便、靈活且功能強(qiáng)大的包裝器。本文就來(lái)和大家聊聊Go中切片和數(shù)組的使用,需要的可以參考一下2023-03-03