gob是Golang包自帶的一個(gè)數(shù)據(jù)結(jié)構(gòu)序列化的編碼/解碼工具。編碼使用Encoder,解碼使用Decoder。一種典型的應(yīng)用場景就是RPC(remote procedure calls)。
gob和json的pack之類的方法一樣,由發(fā)送端使用Encoder對(duì)數(shù)據(jù)結(jié)構(gòu)進(jìn)行編碼。在接收端收到消息之后,接收端使用Decoder將序列化的數(shù)據(jù)變化成本地變量。
package main import ( "bytes" "encoding/gob" "fmt" ) type MsgData struct { X, Y, Z int Name string } var network bytes.Buffer //網(wǎng)絡(luò)傳遞的數(shù)據(jù)載體 func main() { err := senMsg() if err!=nil { fmt.Println("編碼錯(cuò)誤") return } err = revMsg() if err!=nil { fmt.Println("解碼錯(cuò)誤") return } } func senMsg()error { fmt.Print("開始執(zhí)行編碼(發(fā)送端)") enc := gob.NewEncoder(network) sendMsg:=MsgData{3, 4, 5, "jiangzhou"} fmt.Println("原始數(shù)據(jù):",sendMsg) err := enc.Encode(sendMsg) fmt.Println("傳遞的編碼數(shù)據(jù)為:",network) return err } func revMsg()error { var revData MsgData dec:=gob.NewDecoder(network) err:= dec.Decode(revData) //傳遞參數(shù)必須為 地址 fmt.Println("解碼之后的數(shù)據(jù)為:",revData) return err }
1、編碼的數(shù)據(jù)中有空接口類型,傳遞時(shí)賦值的空接口為:基本類型(int、float、string)、切片時(shí),可以不進(jìn)行注冊(cè)。
package main import ( "bytes" "encoding/gob" "fmt" ) type MsgData struct { X, Y, Z int Name string Msg interface{} } var network bytes.Buffer //網(wǎng)絡(luò)傳遞的數(shù)據(jù)載體 func main() { err := senMsg() if err!=nil { fmt.Println("編碼錯(cuò)誤") return } err = revMsg() if err!=nil { fmt.Println("解碼錯(cuò)誤") return } } func senMsg()error { fmt.Print("開始執(zhí)行編碼(發(fā)送端)") enc := gob.NewEncoder(network) s:=make([]string,0) s=append(s, "hello") //sendMsg:=MsgData{3, 4, 5, "jiangzhou",Msg{10001,"hello"}} //sendMsg:=MsgData{3, 4, 5, "jiangzhou",66.66} sendMsg:=MsgData{3, 4, 5, "jiangzhou",s} fmt.Println("原始數(shù)據(jù):",sendMsg) err := enc.Encode(sendMsg) fmt.Println("傳遞的編碼數(shù)據(jù)為:",network) return err } func revMsg()error { var revData MsgData dec:=gob.NewDecoder(network) err:= dec.Decode(revData) //傳遞參數(shù)必須為 地址 fmt.Println("解碼之后的數(shù)據(jù)為:",revData) return err }
編碼的數(shù)據(jù)中有空接口類型,傳遞時(shí)賦值的空接口為:map、struct時(shí),必須進(jìn)行注冊(cè)。
package main import ( "bytes" "encoding/gob" "fmt" ) type MsgData struct { X, Y, Z int Name string Msg interface{} } var network bytes.Buffer //網(wǎng)絡(luò)傳遞的數(shù)據(jù)載體 func main() { err := senMsg() if err!=nil { fmt.Println("編碼錯(cuò)誤") return } err = revMsg() if err!=nil { fmt.Println("解碼錯(cuò)誤") return } } func senMsg()error { fmt.Print("開始執(zhí)行編碼(發(fā)送端)") enc := gob.NewEncoder(network) m:=make(map[int]string) m[10001]="hello" m[10002]="jiangzhou" sendMsg:=MsgData{3, 4, 5, "jiangzhou",m} fmt.Println("原始數(shù)據(jù):",sendMsg) err := enc.Encode(sendMsg) fmt.Println("傳遞的編碼數(shù)據(jù)為:",network) return err } func revMsg()error { var revData MsgData dec:=gob.NewDecoder(network) err:= dec.Decode(revData) //傳遞參數(shù)必須為 地址 fmt.Println("解碼之后的數(shù)據(jù)為:",revData) return err }
Register和RegisterName解決的主要問題是:當(dāng)編解碼中有一個(gè)字段是interface{}(interface{}的賦值為map、結(jié)構(gòu)體時(shí))的時(shí)候需要對(duì)interface{}的可能產(chǎn)生的類型進(jìn)行注冊(cè)。
正確代碼為:
interface{}的賦值為map時(shí):
package main import ( "bytes" "encoding/gob" "fmt" ) type MsgData struct { X, Y, Z int Name string Msg interface{} } var network bytes.Buffer //網(wǎng)絡(luò)傳遞的數(shù)據(jù)載體 func main() { err := senMsg() if err!=nil { fmt.Println("編碼錯(cuò)誤") return } err = revMsg() if err!=nil { fmt.Println("解碼錯(cuò)誤") return } } func senMsg()error { fmt.Print("開始執(zhí)行編碼(發(fā)送端)") enc := gob.NewEncoder(network) m:=make(map[int]string) m[10001]="hello" m[10002]="jiangzhou" gob.Register(map[int]string{}) //TODO:進(jìn)行了注冊(cè) sendMsg:=MsgData{3, 4, 5, "jiangzhou",m} fmt.Println("原始數(shù)據(jù):",sendMsg) err := enc.Encode(sendMsg) fmt.Println("傳遞的編碼數(shù)據(jù)為:",network) return err } func revMsg()error { var revData MsgData dec:=gob.NewDecoder(network) err:= dec.Decode(revData) //傳遞參數(shù)必須為 地址 fmt.Println("解碼之后的數(shù)據(jù)為:",revData) return err }
interface{}的賦值為結(jié)構(gòu)體時(shí):
package main import ( "bytes" "encoding/gob" "fmt" ) type MsgData struct { X, Y, Z int Name string Msg interface{} } var network bytes.Buffer //網(wǎng)絡(luò)傳遞的數(shù)據(jù)載體 func main() { err := senMsg() if err != nil { fmt.Println("編碼錯(cuò)誤",err) return } err = revMsg() if err != nil { fmt.Println("解碼錯(cuò)誤") return } } type Msg struct { Id int Detail string } func senMsg() error { fmt.Print("開始執(zhí)行編碼(發(fā)送端)") enc := gob.NewEncoder(network) gob.Register(Msg{}) //TODO:進(jìn)行了注冊(cè) s:=Msg{10001,"hello jiangzhou"} sendMsg := MsgData{3, 4, 5, "jiangzhou", s} fmt.Println("原始數(shù)據(jù):", sendMsg) err := enc.Encode(sendMsg) fmt.Println("傳遞的編碼數(shù)據(jù)為:", network) return err } func revMsg() error { var revData MsgData dec := gob.NewDecoder(network) err := dec.Decode(revData) //傳遞參數(shù)必須為 地址 fmt.Println("解碼之后的數(shù)據(jù)為:", revData) return err }
注:特別注意:以上代碼中的結(jié)構(gòu)體Msg對(duì)應(yīng)的成員變量名稱首字母一定要大寫,不然會(huì)出現(xiàn):編碼錯(cuò)誤編碼錯(cuò)誤 gob: type main.Msg has no exported fields
這里使用了
gob.Register(Msg{})
告訴系統(tǒng):所有的Interface是有可能為Msg結(jié)構(gòu)的。
在這個(gè)例子中,如果你注釋了gob.Register, 系統(tǒng)會(huì)報(bào)錯(cuò)。
RegisterName是和Register一樣的效果,只是在Register的同時(shí)也為這個(gè)類型附上一個(gè)別名。
補(bǔ)充:GO語音gob包的系列化和反序列化使用和遇到的錯(cuò)誤
encoding/gob包實(shí)現(xiàn)了高效的序列化,特別是數(shù)據(jù)結(jié)構(gòu)較復(fù)雜的,結(jié)構(gòu)體、數(shù)組和切片都被支持。
package main import ( "bytes" "encoding/gob" "fmt" ) //定義一個(gè)結(jié)構(gòu)體 type Person struct { Age int Name string } func main() { p1:=Person{ Age: 18, Name: "貪吃的豬", } //序列化 //這里是儲(chǔ)存的buffer var bufferr bytes.Buffer PerEncod:=gob.NewEncoder(bufferr) //1.創(chuàng)建一個(gè)編碼器 err:=PerEncod.Encode(p1) //編碼 if err != nil { fmt.Println("編碼器 解碼錯(cuò)誤",err) return } //現(xiàn)在buffer就是完成儲(chǔ)存序列化的 fmt.Printf("序列化:buf%x\n",bufferr) //創(chuàng)建一個(gè)空的結(jié)構(gòu)體來接受 p2 :=Person{} //反序列化 PerDecod:=gob.NewDecoder(bytes.NewReader(bufferr.Bytes()))//創(chuàng)建一個(gè)反編碼器 err=PerDecod.Decode(p2) if err != nil { fmt.Println("PerDecod.Decode err:",err) return } fmt.Println("反序列化:",p2) //fmt.Printf("反序列化數(shù)據(jù):string",p2) }
如果是你的結(jié)構(gòu)體的字段是小寫開頭 gob序列化你的結(jié)構(gòu)體的時(shí)候會(huì)找不到字段
如果我把
type Person struct { Age int Name string }
改成
type Person struct { age int name string }
編碼器 解碼錯(cuò)誤 gob: type main.Person has no exported fields
解決方法就是把字段開頭變成大寫
這個(gè)錯(cuò)誤還有一種可能造成的 你定義的結(jié)構(gòu)里面還有一個(gè)結(jié)構(gòu) 2
這個(gè)結(jié)構(gòu)2的字段全部都是小寫開頭
解決方法就是把字段開頭變成大寫
今天是2019年11月2日 11:32 我的一個(gè)改了半天的bug 終于解決
gob在編譯的時(shí)候 如果你的這個(gè)結(jié)構(gòu)體里面包含另一個(gè)結(jié)構(gòu)體
但是另一個(gè)結(jié)構(gòu)體的字段開頭沒有大寫
gob編譯的時(shí)候是不會(huì)報(bào)錯(cuò),他會(huì)不要沒有大寫的字段,
你反序列化的時(shí)候會(huì)發(fā)現(xiàn)這個(gè)字段是nil 空值
我去你碼的
今天是2019年11月4日,今天新的序列化bug出現(xiàn)了
我生成秘鑰對(duì)然后對(duì)密鑰對(duì)進(jìn)行數(shù)據(jù)序列化然后儲(chǔ)存在文件里面
然后錯(cuò)誤提示,在, gob: type not registered for interface: elliptic.p256Curve
其實(shí)gob是可以序列化全部結(jié)構(gòu),但是它不能序列化interface接口
因?yàn)榻涌诘拇笮∈菬o法定義的
密鑰對(duì)的中的公鑰結(jié)構(gòu)體里面一個(gè)字段elliptic.Curve 他是接口
我們把這個(gè)接口進(jìn)行注冊(cè)就行了
gob提供了一個(gè)函數(shù)可以進(jìn)行注冊(cè)
gob.Register(elliptic.P256())
要gob遇到這個(gè)接口的時(shí)候按照elliptic.P256格式進(jìn)行編譯
然后就解決了~
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教。
標(biāo)簽:汕頭 雞西 重慶 蘭州 吐魯番 銅川 欽州 梅河口
巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Golang Gob編碼(gob包的使用詳解)》,本文關(guān)鍵詞 Golang,Gob,編碼,gob,包,的,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。