婷婷综合国产,91蜜桃婷婷狠狠久久综合9色 ,九九九九九精品,国产综合av

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

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

熱門標簽:excel地圖標注分布數據 阿克蘇地圖標注 壽光微信地圖標注 評價高的400電話辦理 百度地圖標注后傳給手機 外呼系統顯本地手機號 涿州代理外呼系統 外呼系統用什么卡 電話機器人軟件免費

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

gob和json的pack之類的方法一樣,由發送端使用Encoder對數據結構進行編碼。在接收端收到消息之后,接收端使用Decoder將序列化的數據變化成本地變量。

基本使用

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name string
}
var network bytes.Buffer //網絡傳遞的數據載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		return
	}
}
 
func senMsg()error {
	fmt.Print("開始執行編碼(發送端)")
 
	enc := gob.NewEncoder(network)
	sendMsg:=MsgData{3, 4, 5, "jiangzhou"}
	fmt.Println("原始數據:",sendMsg)
	err := enc.Encode(sendMsg)
	fmt.Println("傳遞的編碼數據為:",network)
	return  err
}
func revMsg()error {
	var revData MsgData
	dec:=gob.NewDecoder(network)
	err:= dec.Decode(revData) //傳遞參數必須為 地址
	fmt.Println("解碼之后的數據為:",revData)
	return err
}

Register和RegisterName

1、編碼的數據中有空接口類型,傳遞時賦值的空接口為:基本類型(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 //網絡傳遞的數據載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		return
	}
}
 
func senMsg()error {
	fmt.Print("開始執行編碼(發送端)") 
	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("原始數據:",sendMsg)
	err := enc.Encode(sendMsg)
	fmt.Println("傳遞的編碼數據為:",network)
	return  err
}
func revMsg()error {
	var revData MsgData
	dec:=gob.NewDecoder(network)
	err:= dec.Decode(revData) //傳遞參數必須為 地址
	fmt.Println("解碼之后的數據為:",revData)
	return err
}

編碼的數據中有空接口類型,傳遞時賦值的空接口為:map、struct時,必須進行注冊。

package main 
import (
	"bytes"
	"encoding/gob"
	"fmt"
)
 
type MsgData struct {
	X, Y, Z int
	Name string
	Msg interface{}
}
var network bytes.Buffer //網絡傳遞的數據載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		return
	}
}
 
func senMsg()error {
	fmt.Print("開始執行編碼(發送端)")
 
	enc := gob.NewEncoder(network)
 
   m:=make(map[int]string)
	m[10001]="hello"
	m[10002]="jiangzhou"
	sendMsg:=MsgData{3, 4, 5, "jiangzhou",m}
	fmt.Println("原始數據:",sendMsg)
	err := enc.Encode(sendMsg)
	fmt.Println("傳遞的編碼數據為:",network)
	return  err
}
func revMsg()error {
	var revData MsgData
	dec:=gob.NewDecoder(network)
	err:= dec.Decode(revData) //傳遞參數必須為 地址
	fmt.Println("解碼之后的數據為:",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 //網絡傳遞的數據載體
func main() {
	err := senMsg()
	if err!=nil {
		fmt.Println("編碼錯誤")
		return
	}
	err = revMsg()
	if err!=nil {
		fmt.Println("解碼錯誤")
		return
	}
}
 
func senMsg()error {
	fmt.Print("開始執行編碼(發送端)")
 
	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("原始數據:",sendMsg)
	err := enc.Encode(sendMsg)
	fmt.Println("傳遞的編碼數據為:",network)
	return  err
}
func revMsg()error {
	var revData MsgData
	dec:=gob.NewDecoder(network)
	err:= dec.Decode(revData) //傳遞參數必須為 地址
	fmt.Println("解碼之后的數據為:",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 //網絡傳遞的數據載體
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("開始執行編碼(發送端)")
	enc := gob.NewEncoder(network)
	gob.Register(Msg{}) //TODO:進行了注冊
	s:=Msg{10001,"hello jiangzhou"}
	sendMsg := MsgData{3, 4, 5, "jiangzhou", s}
	fmt.Println("原始數據:", sendMsg)
	err := enc.Encode(sendMsg)
	fmt.Println("傳遞的編碼數據為:", network)
	return err
}
func revMsg() error {
	var revData MsgData
	dec := gob.NewDecoder(network)
	err := dec.Decode(revData) //傳遞參數必須為 地址
	fmt.Println("解碼之后的數據為:", revData)
	return err
}

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

這里使用了

gob.Register(Msg{})

告訴系統:所有的Interface是有可能為Msg結構的。

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

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

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

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

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.創建一個編碼器
 err:=PerEncod.Encode(p1) //編碼
 if err != nil {
  fmt.Println("編碼器 解碼錯誤",err)
  return
 }
 //現在buffer就是完成儲存序列化的
 fmt.Printf("序列化:buf%x\n",bufferr)
 
 //創建一個空的結構體來接受
 p2 :=Person{}
 //反序列化
 PerDecod:=gob.NewDecoder(bytes.NewReader(bufferr.Bytes()))//創建一個反編碼器
 err=PerDecod.Decode(p2)
 if err != nil {
  fmt.Println("PerDecod.Decode err:",err)
  return
 }
 fmt.Println("反序列化:",p2)
 //fmt.Printf("反序列化數據: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編譯的時候是不會報錯,他會不要沒有大寫的字段,

你反序列化的時候會發現這個字段是nil 空值

我去你碼的

今天是2019年11月4日,今天新的序列化bug出現了

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

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

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

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

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

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

gob提供了一個函數可以進行注冊

gob.Register(elliptic.P256())

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

然后就解決了~

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

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

標簽:汕頭 雞西 重慶 蘭州 吐魯番 銅川 欽州 梅河口

巨人網絡通訊聲明:本文標題《Golang Gob編碼(gob包的使用詳解)》,本文關鍵詞  Golang,Gob,編碼,gob,包,的,;如發現本文內容存在版權問題,煩請提供相關信息告之我們,我們將及時溝通與處理。本站內容系統采集于網絡,涉及言論、版權與本站無關。
  • 相關文章
  • 下面列出與本文章《Golang Gob編碼(gob包的使用詳解)》相關的同類信息!
  • 本頁收集關于Golang Gob編碼(gob包的使用詳解)的相關信息資訊供網民參考!
  • 推薦文章
    主站蜘蛛池模板: 诸城市| 新乐市| 旺苍县| 南川市| 广昌县| 江津市| 尉犁县| 屯昌县| 喜德县| 乌苏市| 华亭县| 丰县| 南部县| 澳门| 安乡县| 铁岭县| 离岛区| 虎林市| 东至县| 库伦旗| 四会市| 新河县| 如东县| 定兴县| 九江市| 日喀则市| 邢台市| 旅游| 黄平县| 桑日县| 慈利县| 滦南县| 新泰市| 舒兰市| 阜阳市| 临武县| 平凉市| 西充县| 剑河县| 山西省| 房产|