首页 > 其他分享 >go语言序列化和反序列化

go语言序列化和反序列化

时间:2023-07-10 15:46:27浏览次数:48  
标签:语言 err fmt json Printf go 序列化 string

序列化需求

内存中的map、slice、array以及各种对象,如何保存到一个文件中? 如果是自己定义的结构体的实 例,如何保存到一个文件中? 如何从文件中读取数据,并让它们在内存中再次恢复成自己对应的类型的实例?

要设计一套协议,按照某种规则,把内存中数据保存到文件中。文件是一个字节序列,所以必须把数据 转换成字节序列,输出到文件。这就是序列化。 反之,从文件的字节序列恢复到内存并且还是原来的类 型,就是反序列化。

定义

serialization 序列化:将内存中对象存储下来,把它变成一个个字节。转为 二进制数据。

deserialization 反序列化:将文件的一个个字节恢复成内存中对象。从二进制数据中恢复。

序列化保存到文件就是持久化。

字符序列化:JSON、XML等。

二进制序列化:Protocol Buffers、MessagePack等。

可以将数据序列化后持久化,或者网络传输;也可以将从文件中或者网络接收到的字节序列反序列化。可以把数据和二进制序列之间的相互转换称为二进制序列化、反序列化,把数据和字符序列之间的相互转换称为字符序列化、反序列化。

JSON的数据类型

JSON(JavaScript Object Notation, JS 对象标记) 是一种轻量级的数据交换格式。它基于1999年发布的 ES3 (ECMAScript是w3c组织制定的JavaScript规范)的一个子集,采用完全独立于编程语言的文本格 式来存储和表示数据。应该说,目前JSON得到几乎所有浏览器的支持。官网:http://json.org/

JSON是字符串,是文本。JavaScript引擎可以将这种字符串解析为某类型的数据。

object类型

无序键值对的集合

格式: {key1:value1, ... ,keyn:valulen} key必须是一个字符串,需要双引号。 value可以是任意合法的值。

array类型

有序的值的集合

格式:[val1,...,valn]

value类型

双引号引起来的字符串、数值、true和false、null、对象、数组,这些都是值

string类型

由双引号包围起来的任意字符的组合,可以有转义字符。

number类型

有正负,有整数、浮点数。

json包

Go标准库中提供了 encoding/json 包,内部使用了反射技术,效率较为低下。官网https://go.dev/blo g/json

json.Marshal(v any) ([]byte, error),将v序列化成字符序列(本质上也是字节序列),这个过程称 为Encode

json.Unmarshal(data []byte, v any) error,将字符序列data反序列化为v,这个过程称为Decode

基本类型序列化:

package main

import (
    "encoding/json"
    "fmt"
)

func main() {
    // 定义一个any切片类型的变量,将各种类型都填进去
    var data = []any{"牛逼abc", 123, 3.14, true, false, nil,
        map[int]string{},
        [2]int{1, 11}, []string{"a", "b", "c"},
    }
    target := [][]byte{}
    for _, v := range data {
        // 对各种类型的数据进行序列化,需提供任意类型的参数,返回字节序列和
        // error。(v any) ([]byte, error)
        r, err := json.Marshal(v)
        if err != nil {
            fmt.Println(err)
        }
        // 序列化后,会得到字节序列
        fmt.Printf("%T %[1]v ==> %T %[2]v\n", v, r)
        target = append(target, r)
    }
    // target为字节序列类型的切片
    fmt.Println(target)
    fmt.Println("----反序列化----")
    // 对字节序列进行反序列化
    var result any
    for _, v := range target {
        // 提供两个参数,第一个为字节序列,第二个为任意类型,用于接收反序列化后的值,
        // 类似Scan,需要用&来接收,返回error类型。(data []byte, v any) error
        err := json.Unmarshal(v, &result)
        if err != nil {
            fmt.Println(err)
        }
        fmt.Printf("%T %[1]v ==> %T %[2]v\n", v, result)
    }
}
示例1

 下面为上面代码执行结果:

 各种类型数据被序列化就成了字节序列,也可以说转换成了字符串。转换的结果都是字符串,但是这些字符串一旦交给JavaScript引擎,它能把它们转换 成对应的数据类型。

根据反序列化结果,从字符串(字节序列)反序列化为Go的某类型数据。JSON字符串中,数值被转换成了Go的float64类型;true、false转成了bool型; null转成了nil;字符串转成了string;数组转成了[]interface{}

结构体序列化:

package main

import (
    "encoding/json"
    "fmt"
)

type Animal struct {
    // 对于使用 json.Marshal 进行序列化的字段,它们必须是公开的(首字母大写)
    Name string
    Age  int
}

func main() {
    p0 := &Animal{"Tom", 20}
    r, err := json.Marshal(p0)
    if err != nil {
        fmt.Println(err)
    }
    fmt.Printf("%T %[1]v ==> %T %[2]v\n", p0, r)
    fmt.Printf("%v\n", string(r))
    // 对其进行反序列化
    var v0 any
    json.Unmarshal(r, &v0)
    // 反序列化后,成了map了
    fmt.Printf("%T %[1]v\n", v0)
    // 可定义期望的类型的一个变量来进行接收,这样就变成期望的类型了
    var v1 Animal
    json.Unmarshal(r, &v1)
    fmt.Printf("%T %[1]v\n", v1)
}
示例2

下面是上面代码执行结果:

 切片序列化:

package main

import (
    "encoding/json"
    "fmt"
)

func main() {
    s0 := make([]int, 3)
    r, err := json.Marshal(s0)
    if err != nil {
        fmt.Println(err)
    }
    fmt.Printf("%T %[1]v ==> %T %[2]v\n", s0, r)
    var v0 interface{}
    err = json.Unmarshal(r, &v0)
    if err != nil {
        fmt.Println(err)
    }
    fmt.Printf("%T %[1]v\n", v0)
    var v1 []int
    json.Unmarshal(r, &v1)
    fmt.Printf("%T %[1]v\n", v1)
}
示例3

下面是上面代码执行结果:

JSON序列化的Go实现效率较低,由此社区和某些公司提供大量开源的实现,例如easyjson、jsoniter、 sonic等。基本使用方式都兼容官方实现。 

结构体字段标签

结构体的字段可以增加标签tag以供序列化、反序列化时使用

在字段类型后,可以跟反引号引起来的一个标签,用json为key,value用双引号引起来写,key与value直接使用冒号,这个标签中不要加入多余空格,否则语法错误。 例1:Name string `json:"name"`,这个例子序列化后得到的属性名为name,json表示json库使用;双引号内第一个参数用来指定字段转换使用的名称,多个参数使用逗号隔开

例2:Name string `json:"name,omitempty"`,omitempty为序列化时忽略空值,也就是该字段不序列化;空值为false、0、空数组、空切片、空map、空串、nil空指针、nil接口值 空数组、空切片、空串、空map,长度len为0,也就是容器没有元素。

如果使用 - ,该字段将被忽略 Name string `json:"-"`,序列化后没有该字段,反序列化也不会转换该字段,Name string `json:"-,"`,序列化后该字段显示但名为 "-" ,反序列化也会转换该字段

多标签使用空格间隔:Name string `json:"name,omitempty" msgpack:"myname"`

MessagePack

MessagePack是一个基于二进制高效的对象序列化类库,可用于跨语言通信。 它可以像JSON那样,在 许多种语言之间交换结构对象。但是它比JSON更快速也更轻巧。 支持Python、Ruby、Java、C/C++、 Go等众多语言。宣称比Google Protocol Buffers还要快4倍。MessagePack: It's like JSON. but fast and small. (msgpack.org),其基本使用方法和json包类似。

 

package main

import (
    "encoding/json"
    "fmt"

    "github.com/vmihailenco/msgpack/v5"
)

type Animal struct {
    // 标签的各种写法
    Name string `json:"name"`
    Age  int    `msgpack:"a"`
    // 在Color处的json标签对High字段进行空值忽略,结果发现Color字段也有效
    Color  string `json:"High,omitempty" msgpack:"-"`
    Weight string `msgpack:"-"`
    High   float64
}

func main() {
    p0 := &Animal{Name: "Tom", Age: 20}
    r0, _ := json.Marshal(p0)
    r1, _ := msgpack.Marshal(p0)
    fmt.Printf("%T %[1]v ==> %T %[2]v\n", p0, r0)
    fmt.Printf("%T %[1]v ==> %T %[2]v\n", p0, r1)
    fmt.Println(string(r0), string(r1))
    // 反序列化
    var v0 any
    json.Unmarshal(r0, &v0)
    fmt.Printf("%T %[1]v\n", v0)
    msgpack.Unmarshal(r0, &v0)
    fmt.Printf("%T %[1]v\n", v0)
    var v1 Animal
    json.Unmarshal(r0, &v1)
    fmt.Printf("%T %[1]v\n", v1)
    msgpack.Unmarshal(r0, &v1)
    fmt.Printf("%T %[1]v\n", v1)
}
示例4

下面是上面代码执行结果:

 

标签:语言,err,fmt,json,Printf,go,序列化,string
From: https://www.cnblogs.com/caibao666/p/17540196.html

相关文章

  • 关键词:合作博弈 纳什谈判 微网 能源系统 编程语言:matlab 主题:
    关键词:合作博弈纳什谈判微网能源系统编程语言:matlab主题:基于纳什谈判理论的风–光–氢多主体能源系统合作运行方法主要内容:以可再生能源制氢为特征的能源系统将是今后能源互联网建设的重要方向之一。该文针对风–光–氢多主体能源系统的合作运行展开研究。首先,考虑主体间的......
  • 关键词:储能容量优化 储能配置 微网 编程语言:matlab 主
    关键词:储能容量优化储能配置微网编程语言:matlab主题:基于混合整数规划方法的微网电池储能容量优化配置主要内容:本代码目的为实现微电网内电池容量的优化配置,目标函数为配置过程中整体的运行成本最小或者经济效益最大化,约束条件则包括相应的运行约束以及能量平衡约束等等,最后将......
  • 基于FPGA的信号发生器,使用VHDL或Verilog语言进行开发,可以提供相关的仿真和设计说资料
    基于FPGA的信号发生器,使用VHDL或Verilog语言进行开发,可以提供相关的仿真和设计说资料。ID:3150646782307233......
  • Django ORM:最全面的数据库处理指南
    深度探讨DjangoORM的概念、基础使用、进阶操作以及详细解析在实际使用中如何处理数据库操作。同时,我们还讨论了模型深入理解,如何进行CRUD操作,并且深化理解到数据库迁移等高级主题。为了全面解读DjangoORM,我们也讨论了其存在的不足,并对其未来发展进行了展望。这篇文章旨在帮助读者......
  • Django ORM:最全面的数据库处理指南
    深度探讨DjangoORM的概念、基础使用、进阶操作以及详细解析在实际使用中如何处理数据库操作。同时,我们还讨论了模型深入理解,如何进行CRUD操作,并且深化理解到数据库迁移等高级主题。为了全面解读DjangoORM,我们也讨论了其存在的不足,并对其未来发展进行了展望。这篇文章旨在帮助读......
  • R语言 ggplot绘图中添加直线、箭头、注释、矩形等
     001、基础绘图library(gcookbook)#Forthedatasetp<-ggplot(heightweight,aes(x=ageYear,y=heightIn,colour=sex))+geom_point()p##基础绘图 002、添加直线p+geom_hline(yintercept=60)+geom_vline(xinterc......
  • Go优雅的错误处理: 支持错误堆栈, 错误码, 错误链的工具库
    地址:https://github.com/morrisxyang/errors如果觉得有用欢迎Star和PR,有问题请直接提issueerrors简单的支持错误堆栈,错误码,错误链的工具库:支持携带堆栈,嵌套构造错误链支持携带错误码,方便接口返回支持自定义堆栈打印深度和错误链打印格式使用......
  • 测试工作繁琐,想要提高效率?来试试这款测试平台RunnerGo
    RunnerGo是什么?RunnerGo是一款轻量级、全栈式的测试平台,支持接口管理、场景管理、性能测试、自动化测试等功能。与市面上的性能测试工具不同的是RunnerGo基于go语言研发,更加轻量级。RunnerGo有什么优势?做软件测试的同学在工作时应该都碰到过这种情况:接口管理、接口测试用postm......
  • c语言学习2
    一、数据类型为什么要对数据进行分类?1、现实中的数据就是自带类别属性的2、对数据进行分类可以节约内存存储空间、提高运行速度存储空间的单位:Bit比特存储一个二进制位,只能存储0或者1,计算机存储数据的最小单位Byte字节存储八个二进制位,计算机存储数据的基本单位Kb102......
  • Django中migrate 出现问题
    问题:不小心删掉了数据库中的表格,重新迁移时出现如下问题 解决办法:1.在数据库中新建dashboard_piechart表2.运行pymanage.pymigratedashboardzero3.运行pymanage.pymigratedashboard查看数据库,新建了Model中的表格......