golang中使用proto3協(xié)議導(dǎo)致的空值字段不顯示的問(wèn)題處理方案
最近在使用grpc協(xié)議的時(shí)候,由于采用的是Proto3協(xié)議,在查找記錄信息的時(shí)候,由于某些字段會(huì)有默認(rèn)空值,導(dǎo)致在通過(guò)協(xié)議調(diào)用后,返回的json結(jié)構(gòu)中并沒(méi)有這些字段,雖然作為前端使用沒(méi)有太大的問(wèn)題,但是在更多的使用場(chǎng)景中,我們更需要知道該服務(wù)返回的確切字段,以便于能夠做相應(yīng)處理,尤其是編譯型語(yǔ)言
具體的使用出現(xiàn)場(chǎng)景如下
type MemberResponse struct { Id int32 `json "id"` Phone string `json "phone"` Age int8 `json "age"` } //獲取用戶信息的接口 func (m *Member) GetMember(req *proto.MemberRequest, resp * proto.MemberResponse) error { resp.Phone = "15112810201" resp.Id = 12 return nil }
當(dāng)通過(guò)api調(diào)用該微服務(wù)后,在proto3協(xié)議下,會(huì)返回如下結(jié)果:
{ "phone" : "15112810201", "id" : 12 }
此時(shí)就會(huì)出現(xiàn)空值的Age字段沒(méi)有返回到對(duì)應(yīng)的json結(jié)構(gòu)中,而這樣在某些情況下對(duì)前端也是不太友好的,尤其是APP客戶端,更需要明確的json響應(yīng)字段結(jié)構(gòu),那么我們可以怎么處理這個(gè)問(wèn)題呢,經(jīng)過(guò)研究和網(wǎng)上的解答,有兩種辦法:
- 直接修改經(jīng)過(guò)protoc生成的member.pb.go文件代碼,刪除掉不希望被忽略的字段tag標(biāo)簽中的omitempty即可,但是*.pb.go一般我們不建議去修改它,而且我們會(huì)經(jīng)常去調(diào)整grpc微服務(wù)協(xié)議中的方法或者字段內(nèi)容,這樣每次protoc之后,都需要我們?nèi)バ薷?,這顯然是不太現(xiàn)實(shí)的,因此就有了第二種辦法;
- 通過(guò)grpc官方庫(kù)中的jsonpb來(lái)實(shí)現(xiàn),官方在它的設(shè)定中有一個(gè)結(jié)構(gòu)體用來(lái)實(shí)現(xiàn)protoc buffer轉(zhuǎn)換為JSON結(jié)構(gòu),并可以根據(jù)字段來(lái)配置轉(zhuǎn)換的要求,結(jié)構(gòu)體如下:
// Marshaler is a configurable object for converting between // protocol buffer objects and a JSON representation for them. type Marshaler struct { // 是否將枚舉值設(shè)定為整數(shù),而不是字符串類型. EnumsAsInts bool // 是否將字段值為空的渲染到JSON結(jié)構(gòu)中 EmitDefaults bool //縮進(jìn)每個(gè)級(jí)別的字符串 Indent string //是否使用原生的proto協(xié)議中的字段 OrigName bool }
了解了這個(gè)結(jié)構(gòu)體之后呢,我們就開(kāi)始對(duì)應(yīng)的使用辦法:
通過(guò)Marshaler結(jié)構(gòu)體的Marshal方法,實(shí)現(xiàn)了將proto響應(yīng)的內(nèi)容轉(zhuǎn)化為buffer,最終輸出為JSON結(jié)構(gòu),從而實(shí)現(xiàn)了空值字段的返回
import ( member "proto/member" ) var jsonpbMarshaler *jsonpb.Marshaler func queryHandler(req *http.Requst, resp http.ResponseWriter){ var ( _buffer bytes.Buffer ) memberResponse, err := member.GetMember(context.TODO(), &member.MemberRequest{}) //調(diào)用此方法實(shí)現(xiàn)轉(zhuǎn)換 jsonpbMarshaler.Marshal(&_buffer, memberResponse) jsonCnt := _buffer.Bytes() resp.Header().Set('Content-Type', 'application/json') resp.Write(jsonCnt) return } func main(){ jsonpbMarshaler = &jsonpb.Marshaler{ EnumsAsInts : true, EmitDefaults: true, OrigName : true } //其他http處理代碼塊 }
寫這篇文章只是為了吧這個(gè)解決方案分享給大家,希望對(duì)有此疑問(wèn)的朋友們有所幫助。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Golang Map value不可尋址使用指針類型代替示例詳解
這篇文章主要為大家介紹了Golang Map value不可尋址使用指針類型代替示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-11-11基于go+vue實(shí)現(xiàn)的golang每日新聞數(shù)據(jù)瀏覽與檢索平臺(tái)(推薦)
gonews是基于 go+vue 實(shí)現(xiàn)的golang每日新聞瀏覽與檢索平臺(tái),本文通過(guò)實(shí)例代碼給大家講解,介紹的非常詳細(xì),具有參考借鑒價(jià)值,需要的朋友參考下吧2018-01-01go?REST?API設(shè)計(jì)模式和反模式示例解析
在這篇文章中,我們將探討一些常見(jiàn)的REST?API設(shè)計(jì)模式和開(kāi)發(fā)者應(yīng)該注意的反模式,我們還將提供Golang和Open?API?Schema的代碼片段來(lái)幫助說(shuō)明這些概念,有需要的朋友可以借鑒參考下2023-09-09源碼剖析Golang中singleflight的應(yīng)用
這篇文章主要為大家詳細(xì)介紹了如何利用singleflight來(lái)避免緩存擊穿,并剖析singleflight包的源碼實(shí)現(xiàn)和工作原理,感興趣的可以了解下2024-03-03Go語(yǔ)言實(shí)現(xiàn)服務(wù)端消息接收和發(fā)送
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言實(shí)現(xiàn)服務(wù)端消息接收和發(fā)送功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-07-07簡(jiǎn)單聊一聊Go語(yǔ)言中的數(shù)組和切片
數(shù)組和切片由于語(yǔ)法十分相似,在使用中容易混淆,要認(rèn)真區(qū)分,下面這篇文章主要給大家介紹了關(guān)于Go語(yǔ)言中數(shù)組和切片的相關(guān)資料,需要的朋友可以參考下2021-07-07go 迭代string數(shù)組操作 go for string[]
這篇文章主要介紹了go 迭代string數(shù)組操作 go for string[],具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-12-12Go?slice切片make生成append追加copy復(fù)制示例
這篇文章主要為大家介紹了Go使用make生成切片、使用append追加切片元素、使用copy復(fù)制切片使用示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06