【go从零单排】JSON序列化和反序列化

embedded/2024/11/14 6:12:38/

挪威特罗姆瑟夜景

🌈Don’t worry , just coding!
内耗与overthinking只会削弱你的精力,虚度你的光阴,每天迈出一小步,回头时发现已经走了很远。

📗概念

在 Go 语言中,处理 JSON 数据主要依赖于 encoding/json 包。这个包提供了编码(序列化)和解码(反序列化)JSON 数据的功能。

💻代码

Example

package main//导入了 encoding/json(用于处理 JSON 数据)、fmt(用于格式化输出)和 os(用于操作系统功能)包。
import ("encoding/json""fmt""os"
)// 定义了两个结构体 response1 和 response2。
// response2 使用了结构体标签(tags),指定了 JSON 字段名,这样在序列化时可以控制字段名称
type response1 struct {Page   int//首字母必须大写才能导出字段Fruits []string
}type response2 struct {Page   int      `json:"page"`Fruits []string `json:"fruits"`
}func main() {//使用 json.Marshal 将其转换为 JSON 格式的字节切片,并转为字符串输出。bolB, _ := json.Marshal(true)fmt.Println(string(bolB)) // 输出: trueintB, _ := json.Marshal(1)fmt.Println(string(intB)) // 输出: 1fltB, _ := json.Marshal(2.34)fmt.Println(string(fltB)) // 输出: 2.34strB, _ := json.Marshal("gopher")fmt.Println(string(strB)) // 输出: "gopher"slcD := []string{"apple", "peach", "pear"}slcB, _ := json.Marshal(slcD)fmt.Println(string(slcB)) // 输出: ["apple","peach","pear"]mapD := map[string]int{"apple": 5, "lettuce": 7}mapB, _ := json.Marshal(mapD)fmt.Println(string(mapB)) // 输出: {"apple":5,"lettuce":7}res1D := &response1{Page:   1,Fruits: []string{"apple", "peach", "pear"}}res1B, _ := json.Marshal(res1D)fmt.Println(string(res1B)) // 输出: {"Page":1,"Fruits":["apple","peach","pear"]}res2D := &response2{Page:   1,Fruits: []string{"apple", "peach", "pear"}}res2B, _ := json.Marshal(res2D)fmt.Println(string(res2B)) // 输出: {"page":1,"fruits":["apple","peach","pear"]}byt := []byte(`{"num":6.13,"strs":["a","b"]}`)var dat map[string]interface{}if err := json.Unmarshal(byt, &dat); err != nil {panic(err)}fmt.Println(dat) // 输出: map[num:6.13 strs:[a b]]num := dat["num"].(float64)//num是float的类型,这里需要手动转化fmt.Println(num)strs := dat["strs"].([]interface{})str1 := strs[0].(string)fmt.Println(str1)str := `{"page": 1, "fruits": ["apple", "peach"]}`res := response2{}//[]byte(str) 将 JSON 字符串 str 转换为字节切片,以便 Unmarshal 函数可以处理。//&res 是一个指向 res 的指针,表示解码后的数据将存储在 res 中。json.Unmarshal([]byte(str), &res)fmt.Println(res)           // 输出: {1 [apple peach]}fmt.Println(res.Fruits[0]) // 输出: apple//使用 json.NewEncoder 创建一个 JSON 编码器,将数据直接编码到标准输出流(os.Stdout)。enc := json.NewEncoder(os.Stdout)d := map[string]int{"apple": 5, "lettuce": 7}enc.Encode(d) // 输出: {"apple":5,"lettuce":7}
}//输出
//true
//1
//2.34
//"gopher"
//["apple","peach","pear"]
//{"apple":5,"lettuce":7}
//{"Page":1,"Fruits":["apple","peach","pear"]}
//{"page":1,"fruits":["apple","peach","pear"]}
//map[num:6.13 strs:[a b]]
//6.13
//a
//{1 [apple peach]}
//apple
//{"apple":5,"lettuce":7}

序列化

将 Go 数据结构转换为 JSON 格式的字符串。

package mainimport ("encoding/json""fmt"
)func main() {// 示例数据data := map[string]interface{}{"name": "Alice","age":  30,"hobbies": []string{"reading", "traveling"},}// 编码为 JSONjsonData, err := json.Marshal(data)if err != nil {fmt.Println("Error encoding JSON:", err)return}fmt.Println(string(jsonData)) // 输出: {"age":30,"hobbies":["reading","traveling"],"name":"Alice"}
}

反序列化

将 JSON 格式的字符串转换为 Go 数据结构。

package mainimport ("encoding/json""fmt"
)func main() {jsonStr := `{"name": "Alice", "age": 30, "hobbies": ["reading", "traveling"]}`var data map[string]interface{}// 解码 JSONerr := json.Unmarshal([]byte(jsonStr), &data)if err != nil {fmt.Println("Error decoding JSON:", err)return}fmt.Println(data) // 输出: map[age:30 hobbies:[reading traveling] name:Alice]
}

切片序列化

Go 的切片可以直接映射到 JSON 数组。

func main() {fruits := []string{"apple", "banana", "cherry"}jsonData, err := json.Marshal(fruits)if err != nil {fmt.Println("Error encoding JSON:", err)return}fmt.Println(string(jsonData)) // 输出: ["apple","banana","cherry"]
}

使用 JSON 标签

通过结构体标签自定义 JSON 字段名。

type Person struct {Name    string   `json:"name"`Age     int      `json:"age"`Hobbies []string `json:"hobbies"`
}

序列化结构体

package mainimport ("encoding/json""fmt"
)
//定义结构体
type Person struct {Name    string   `json:"name"`Age     int      `json:"age"`Hobbies []string `json:"hobbies"`
}func main() {person := Person{Name:    "Alice",Age:     30,Hobbies: []string{"reading", "traveling"},}//序列化结构体jsonData, err := json.Marshal(person)if err != nil {fmt.Println("Error encoding JSON:", err)return}fmt.Println(string(jsonData)) // 输出: {"age":30,"hobbies":["reading","traveling"],"name":"Alice"}
}

反序列化结构体

func main() {jsonStr := `{"name": "Alice", "age": 30, "hobbies": ["reading", "traveling"]}`var person Person//[]byte(jsonStr) 将一个字符串(jsonStr)转换为字节切片。这是因为 json.Unmarshal 需要一个字节切片作为输入参数,而不是字符串。err := json.Unmarshal([]byte(jsonStr), &person)if err != nil {fmt.Println("Error decoding JSON:", err)return}fmt.Println(person) // 输出: {Alice 30 [reading traveling]}
}
  • 在 Go 中,json.Unmarshal 函数用于将 JSON 格式的字节切片解码为 Go 数据结构。这里的 []byte 是一个字节切片类型,表示 JSON 数据的原始字节序列。
  • 字节切片:[]byte 是 Go 中的一个基本数据类型,表示一个字节的动态数组。每个字节是一个 8 位的无符号整数,通常用于表示原始数据,如文本、二进制数据等。
  • JSON 字符串:在 JSON 数据的上下文中,JSON 字符串通常是以 UTF-8 编码的文本。要将 JSON 字符串转换为 Go 数据结构,首先需要将其转换为字节切片。

🔍理解

  • 序列化是把go的数据结构转为json格式,用json.Marshal(data)
  • 反序列化是把json格式转为go的数据格式,用json.Unmarshal

💡 Tips小知识点

错误处理

在编码和解码过程中,建议检查错误,以确保数据的正确性。

if err != nil {fmt.Println("Error:", err)
}

JSON 编码到输出流

使用 json.Encoder 将 JSON 数据直接写入到输出流(如标准输出)。

enc := json.NewEncoder(os.Stdout)
data := map[string]int{"apple": 5, "banana": 2}
enc.Encode(data) // 输出: {"apple":5,"banana":2}

💪无人扶我青云志,我自踏雪至山巅。
在这里插入图片描述


http://www.ppmy.cn/embedded/137428.html

相关文章

C 语言函数

目录 1.简介 2.main() 3.参数的传值引用 4.函数指针 5.函数原型 6.exit() 7.函数说明符 7.1 extern 说明符 7.2 static 说明符 7.3 const 说明符 8.可变参数 1.简介 函数是一段可以重复执行的代码。它可以接受不同的参数,完成对应的操作。下面的例子就…

VLC-QT----Linux编译并运行示例

linux:ubuntu 16.04 qt:5.13.2 总体安装步骤 下载安装,编译 下载源码仓库,下载cmake,新建一个build文件夹,cd进去,执行代码 cmake .. -DCMAKE_BUILD_TYPEDebug 遇到报错,没有qt5Coreconfig,运行 sudo apt-get install qtdeclarative5-dev进行安装 遇到报错 Could not fi…

MySQL与Oracle对比及区别

一、比较 1、MySQL的特点 性能卓越,服务稳定,很少出现异常宕机; 开放源代码无版本制约,自主性及使用成本低; 历史悠久,社区和用户非常活跃,遇到问题及时寻求帮助; 软件体积小&#…

巧妙注入的奥秘:在 Spring 中优雅地使用 List 和 Map

文章目录 一、注入 List:同类项,一次拿下二、注入 Map:当键值对遇上多态三、进阶:使用 Qualifier 灵活注入四、总结总结推荐阅读文章 在 Spring 框架里,我们经常需要将不同的组件组织到集合中,比如在注入多…

边缘计算在工业互联网中的应用

💓 博客主页:瑕疵的CSDN主页 📝 Gitee主页:瑕疵的gitee主页 ⏩ 文章专栏:《热点资讯》 边缘计算在工业互联网中的应用 边缘计算在工业互联网中的应用 边缘计算在工业互联网中的应用 引言 边缘计算概述 定义与原理 发展…

部署zabbix遇到问题: cannot find a valid baseurl for repo:centos-sclo-rh/x86 64 怎么解决 ?

安装 Zabbix 前端包,提示cannot find a valid baseurl for repo:centos-sclo-rh/x86 64 安装zabbix前端包 # yum install zabbix-web-mysql-scl zabbix-apache-conf-scl 解决办法: 原因是:CentOS7的SCL源在2024年6月30日停止维护…

odoo-040 odoo17前端的js方法调用后端py方法action报错

文章目录 问题描述梳理写法xml写法前端方法后端action的写法 错误解释 问题描述 在前端的kanban视图上添加了几个自定义按钮,按钮点击可以跳转到对应的tree视图,在写按钮调用方法的时候报错如下: 前端调用后端action报错: actio…

快递物流查询API接口如何用PHP调用

在现代商业中,供应链的协同运作至关重要。 快递物流查询API接口可以实现供应商、电商平台、物流企业和消费者之间的信息无缝对接,各方能够及时获取快递物流信息,从而更好地协调生产、销售和配送等环节,提高整个供应链的效率和效益…