佳木斯湛栽影视文化发展公司

主頁 > 知識(shí)庫 > Golang Gob編碼(gob包的使用詳解)

Golang Gob編碼(gob包的使用詳解)

熱門標(biāo)簽:鐵路電話系統(tǒng) 呼叫中心市場需求 網(wǎng)站排名優(yōu)化 AI電銷 百度競價(jià)排名 Linux服務(wù)器 地方門戶網(wǎng)站 服務(wù)外包

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
}

Register和RegisterName

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)
}

系列化和反系列化的常見的錯(cuò)誤

如果是你的結(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ò)誤或未考慮完全的地方,望不吝賜教。

您可能感興趣的文章:
  • 基于golang uint8、int8與byte的區(qū)別說明
  • golang 監(jiān)聽服務(wù)的信號(hào),實(shí)現(xiàn)平滑啟動(dòng),linux信號(hào)說明詳解
  • golang 實(shí)現(xiàn)時(shí)間戳和時(shí)間的轉(zhuǎn)化
  • golang如何獲得一個(gè)變量的類型
  • golang 如何獲取文件夾下面的文件列表
  • golang 如何實(shí)現(xiàn)HTTP代理和反向代理
  • Golang實(shí)現(xiàn)http文件上傳小功能的案例
  • golang值類型轉(zhuǎn)換成[]uint8類型的操作

標(biāo)簽:仙桃 蘭州 黃山 湖南 銅川 湘潭 崇左 衡水

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Golang Gob編碼(gob包的使用詳解)》,本文關(guān)鍵詞  ;如發(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)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266
    石泉县| 耒阳市| 溧水县| 延津县| 高要市| 西乌珠穆沁旗| 内江市| 垦利县| 二连浩特市| 新宾| 安乡县| 出国| 梅河口市| 丘北县| 涞水县| 绥芬河市| 威信县| 平潭县| 霍林郭勒市| 九龙县| 大足县| 乐陵市| 宾阳县| 鲁山县| 兰考县| 永修县| 临西县| 宁明县| 化德县| 广元市| 河曲县| 宜宾市| 霍邱县| 鲜城| 宁德市| 灵山县| 福贡县| 连山| 尼木县| 大足县| 嘉禾县|