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

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

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

gob是Golang包自帶的一個數(shù)據(jù)結構序列化的編碼/解碼工具。編碼使用Encoder,解碼使用Decoder。一種典型的應用場景就是RPC(remote procedure calls)。

gob和json的pack之類的方法一樣,由發(fā)送端使用Encoder對數(shù)據(jù)結構進行編碼。在接收端收到消息之后,接收端使用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)絡傳遞的數(shù)據(jù)載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		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ù)中有空接口類型,傳遞時賦值的空接口為:基本類型(int、float、string)、切片時,可以不進行注冊。

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name string
	Msg interface{}
}
var network bytes.Buffer //網(wǎng)絡傳遞的數(shù)據(jù)載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		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ù)中有空接口類型,傳遞時賦值的空接口為:map、struct時,必須進行注冊。

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name string
	Msg interface{}
}
var network bytes.Buffer //網(wǎng)絡傳遞的數(shù)據(jù)載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		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解決的主要問題是:當編解碼中有一個字段是interface{}(interface{}的賦值為map、結構體時)的時候需要對interface{}的可能產生的類型進行注冊。

正確代碼為:

interface{}的賦值為map時:

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name string
	Msg interface{}
}
var network bytes.Buffer //網(wǎng)絡傳遞的數(shù)據(jù)載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		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:進行了注冊
	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{}的賦值為結構體時:

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name    string
	Msg     interface{}
}
 
var network bytes.Buffer //網(wǎng)絡傳遞的數(shù)據(jù)載體
func main() {
	err := senMsg()
	if err != nil {
		fmt.Println("編碼錯誤",err)
		return
	}
	err = revMsg()
	if err != nil {
		fmt.Println("解碼錯誤")
		return
	}
}
 
type Msg struct {
	Id     int
	Detail string
}
 
func senMsg() error {
	fmt.Print("開始執(zhí)行編碼(發(fā)送端)")
	enc := gob.NewEncoder(network)
	gob.Register(Msg{}) //TODO:進行了注冊
	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
}

注:特別注意:以上代碼中的結構體Msg對應的成員變量名稱首字母一定要大寫,不然會出現(xiàn):編碼錯誤編碼錯誤 gob: type main.Msg has no exported fields

這里使用了

gob.Register(Msg{})

告訴系統(tǒng):所有的Interface是有可能為Msg結構的。

在這個例子中,如果你注釋了gob.Register, 系統(tǒng)會報錯。

RegisterName是和Register一樣的效果,只是在Register的同時也為這個類型附上一個別名。

補充:GO語音gob包的系列化和反序列化使用和遇到的錯誤

encoding/gob包實現(xiàn)了高效的序列化,特別是數(shù)據(jù)結構較復雜的,結構體、數(shù)組和切片都被支持。

package main
 
import (
 "bytes"
 "encoding/gob"
 "fmt"
)
//定義一個結構體
type Person struct {
 Age int
 Name string
}
 
func main() {
 p1:=Person{
  Age:  18,
  Name: "貪吃的豬",
 }
 //序列化
 //這里是儲存的buffer
 var bufferr bytes.Buffer
 PerEncod:=gob.NewEncoder(bufferr) //1.創(chuàng)建一個編碼器
 err:=PerEncod.Encode(p1) //編碼
 if err != nil {
  fmt.Println("編碼器 解碼錯誤",err)
  return
 }
 //現(xiàn)在buffer就是完成儲存序列化的
 fmt.Printf("序列化:buf%x\n",bufferr)
 
 //創(chuàng)建一個空的結構體來接受
 p2 :=Person{}
 //反序列化
 PerDecod:=gob.NewDecoder(bytes.NewReader(bufferr.Bytes()))//創(chuàng)建一個反編碼器
 err=PerDecod.Decode(p2)
 if err != nil {
  fmt.Println("PerDecod.Decode err:",err)
  return
 }
 fmt.Println("反序列化:",p2)
 //fmt.Printf("反序列化數(shù)據(jù):string",p2)
}

系列化和反系列化的常見的錯誤

如果是你的結構體的字段是小寫開頭 gob序列化你的結構體的時候會找不到字段

如果我把

type Person struct {
    Age int
    Name string
}

改成

type Person struct {
    age int
    name string
}

編碼器 解碼錯誤 gob: type main.Person has no exported fields

解決方法就是把字段開頭變成大寫

這個錯誤還有一種可能造成的 你定義的結構里面還有一個結構 2

這個結構2的字段全部都是小寫開頭

解決方法就是把字段開頭變成大寫

今天是2019年11月2日 11:32 我的一個改了半天的bug 終于解決

gob在編譯的時候 如果你的這個結構體里面包含另一個結構體

但是另一個結構體的字段開頭沒有大寫

gob編譯的時候是不會報錯,他會不要沒有大寫的字段,

你反序列化的時候會發(fā)現(xiàn)這個字段是nil 空值

我去你碼的

今天是2019年11月4日,今天新的序列化bug出現(xiàn)了

我生成秘鑰對然后對密鑰對進行數(shù)據(jù)序列化然后儲存在文件里面

然后錯誤提示,在, gob: type not registered for interface: elliptic.p256Curve

其實gob是可以序列化全部結構,但是它不能序列化interface接口

因為接口的大小是無法定義的

密鑰對的中的公鑰結構體里面一個字段elliptic.Curve 他是接口

我們把這個接口進行注冊就行了

gob提供了一個函數(shù)可以進行注冊

gob.Register(elliptic.P256())

要gob遇到這個接口的時候按照elliptic.P256格式進行編譯

然后就解決了~

以上為個人經(jīng)驗,希望能給大家一個參考,也希望大家多多支持腳本之家。如有錯誤或未考慮完全的地方,望不吝賜教。

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

標簽:仙桃 蘭州 黃山 湖南 銅川 湘潭 崇左 衡水

巨人網(wǎng)絡通訊聲明:本文標題《Golang Gob編碼(gob包的使用詳解)》,本文關鍵詞  ;如發(fā)現(xiàn)本文內容存在版權問題,煩請?zhí)峁┫嚓P信息告之我們,我們將及時溝通與處理。本站內容系統(tǒng)采集于網(wǎng)絡,涉及言論、版權與本站無關。
  • 相關文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266