欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

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

 更新時間:2021年05月07日 10:17:04   作者:cqu_jiangzhou  
這篇文章主要介紹了Golang Gob編碼(gob包的使用詳解),具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧

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

gob和json的pack之類的方法一樣,由發(fā)送端使用Encoder對數(shù)據(jù)結(jié)構(gòu)進行編碼。在接收端收到消息之后,接收端使用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、結(jié)構(gòu)體時)的時候需要對interface{}的可能產(chǎn)生的類型進行注冊。

正確代碼為:

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{}的賦值為結(jié)構(gòu)體時:

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
}

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

這里使用了

gob.Register(Msg{})

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

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

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

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

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

package main
 
import (
 "bytes"
 "encoding/gob"
 "fmt"
)
//定義一個結(jié)構(gòu)體
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)建一個空的結(jié)構(gòu)體來接受
 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)
}

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

如果是你的結(jié)構(gòu)體的字段是小寫開頭 gob序列化你的結(jié)構(gòu)體的時候會找不到字段

如果我把

type Person struct {
    Age int
    Name string
}

改成

type Person struct {
    age int
    name string
}

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

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

這個錯誤還有一種可能造成的 你定義的結(jié)構(gòu)里面還有一個結(jié)構(gòu) 2

這個結(jié)構(gòu)2的字段全部都是小寫開頭

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

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

gob在編譯的時候 如果你的這個結(jié)構(gòu)體里面包含另一個結(jié)構(gòu)體

但是另一個結(jié)構(gòu)體的字段開頭沒有大寫

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

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

我去你碼的

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

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

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

其實gob是可以序列化全部結(jié)構(gòu),但是它不能序列化interface接口

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

密鑰對的中的公鑰結(jié)構(gòu)體里面一個字段elliptic.Curve 他是接口

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

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

gob.Register(elliptic.P256())

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

然后就解決了~

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

相關文章

  • go語言通過反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對應的操作

    go語言通過反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對應的操作

    這篇文章主要介紹了go語言通過反射創(chuàng)建結(jié)構(gòu)體、賦值、并調(diào)用對應的操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2021-05-05
  • go中控制goroutine數(shù)量的方法

    go中控制goroutine數(shù)量的方法

    這篇文章主要介紹了go中控制goroutine數(shù)量的方法,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2021-05-05
  • Golang利用位運算實現(xiàn)為程序加速

    Golang利用位運算實現(xiàn)為程序加速

    這篇文章主要為大家詳細介紹了如何在Golang中利用位運算實現(xiàn)為程序加速功能,文中的示例代碼講解詳細,感興趣的小伙伴可以了解一下
    2022-08-08
  • Go語言設計模式之結(jié)構(gòu)型模式

    Go語言設計模式之結(jié)構(gòu)型模式

    本文主要聚焦在結(jié)構(gòu)型模式(Structural Pattern)上,其主要思想是將多個對象組裝成較大的結(jié)構(gòu),并同時保持結(jié)構(gòu)的靈活和高效,從程序的結(jié)構(gòu)上解決模塊之間的耦合問題
    2021-06-06
  • Go?Fiber快速搭建一個HTTP服務器

    Go?Fiber快速搭建一個HTTP服務器

    Fiber?是一個?Express?啟發(fā)?web?框架基于?fasthttp?,最快?Go?的?http?引擎,這篇文章主要介紹了Go?Fiber快速搭建一個HTTP服務器,需要的朋友可以參考下
    2023-06-06
  • Golang Copier入門到入坑探究

    Golang Copier入門到入坑探究

    這篇文章主要為大家介紹了Golang Copier入門到入坑探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-11-11
  • 一文帶你掌握go中的異常處理

    一文帶你掌握go中的異常處理

    代碼在執(zhí)行的過程中可能因為一些邏輯上的問題而出現(xiàn)錯誤,我們稱之為異常,那么如何處理這些異常呢,下面小編就來和大家詳細聊聊golang中的異常處理機制吧
    2023-09-09
  • Go使用Viper庫讀取YAML配置文件的示例代碼

    Go使用Viper庫讀取YAML配置文件的示例代碼

    Viper是適用于Go應用程序的完整配置解決方案,它被設計用于在應用程序中工作,并且可以處理所有類型的配置需求和格式,本文給大家介紹了Go使用Viper庫讀取YAML配置文件的方法,需要的朋友可以參考下
    2024-05-05
  • golang字符串本質(zhì)與原理詳解

    golang字符串本質(zhì)與原理詳解

    這篇文章主要介紹了golang字符串本質(zhì)與原理詳解,golang中的字符串指的是所有8比特位字節(jié)字符串的集合,通常是UTF-8編碼的文本,更多相關內(nèi)容需要的小伙伴可以參考一下
    2022-06-06
  • GO語言中通道和sync包的使用教程分享

    GO語言中通道和sync包的使用教程分享

    這篇文章主要為大家詳細介紹了Go語言中通道和sync包的相關資料,文中的示例代碼講解詳細,對我們學習Go語言有一定的幫助,需要的可以參考一下
    2023-02-02

最新評論