Go 语言 JSON 标准库的使用
来源:脚本之家
时间:2022-12-23 21:31:04 230浏览 收藏
本篇文章主要是结合我之前面试的各种经历和实战开发中遇到的问题解决经验整理的,希望这篇《Go 语言 JSON 标准库的使用》对你有很大帮助!欢迎收藏,分享给更多的需要的朋友学习~
Go 语言中的 encoding/json 库提供了复杂的将 Go 中各种类型与JSON格式之间转换的功能, 我们主要使用以下几个功能:
- 将一个切片、结构体或字典序列化成 JSON 格式的字符串【字节流】。
- 将一个 JSON 格式的字符串【字节流】反序列化成一个切片、结构体或字典。
序列化
序列化使用 json 库中的Marshal
函数:
func Marshal(v interface{}) ([]byte, error)
1. 结构体序列化
比如使用以下的结构体表示一部电影:
type Movie struct { Title string Year int `json:"released"` Color bool `json:"color,omitempty"` Actors []string }
定义里类型后面跟的字符串 json:"released"
和json:"color,omitempty
,称为 field tags,它告诉 json 库在执行序列化时的一些规则:
json:"released"
使得在序列化后对应的名字为"released",而不是"Year"。json:"color,omitempty"
使得如果 Color 成员的值为false
,那么就忽略它,不对它进行序列化。
没有 field tags 时进行序列化的一些规则:
- 如果结构体成员的名字不是以大写字母开头,则不对它进行序列化。
- 如果结构体成员的名字是以大写字母开头,则序列化后的名字就是成员名。
进行序列化的代码如下:
movie := Movie{ Title: "Casablanca", Year: 1942, Color: false, Actors: []string{"Humphrey Bogart", "Ingrid Bergman"}, } data, err := json.Marshal(movie) if err != nil { log.Fatalf("JSON marshaling failed: %s", err) } fmt.Printf("%s\n", data)
输出:
{"Title":"Casablanca","released":1942,"Actors":["Humphrey Bogart","Ingrid Bergman"]}
2. 字典序列化
一个字典要想序列化成- JSON 格式,它的 key 必须是字符串。
以下是一个例子:
info := map[string]int{ "width": 1280, "height": 720, } data, err := json.MarshalIndent(info, "", " ") if err != nil { log.Fatalf("JSON marshaling failed: %s", err) } fmt.Printf("%s\n", data)
输出:
{
"height": 720,
"width": 1280
}
这里我们使用MarshalIndent
函数,使得输出后的 JSON 格式更易阅读。序列化后的名字就是字典中 key 的名称。
3. 切片序列化
直接看一个例子:
type Movie struct { Title string Year int `json:"released"` Color bool `json:"color,omitempty"` Actors []string } var movies = []Movie{ { Title: "Casablanca", Year: 1942, Color: false, Actors: []string{"Humphrey Bogart", "Ingrid Bergman"}, }, { Title: "Cool Hand Luke", Year: 1967, Color: true, Actors: []string{"Paul Newman"}, }, { Title: "Bullitt", Year: 1968, Color: true, Actors: []string{"Steve McQueen", "Jacqueline Bisset"}, }, } data, err := json.MarshalIndent(movies, "", " ") if err != nil { log.Fatalf("JSON marshaling failed: %s", err) } fmt.Printf("%s\n", data)
输出:
[
{
"Title": "Casablanca",
"released": 1942,
"Actors": [
"Humphrey Bogart",
"Ingrid Bergman"
]
},
{
"Title": "Cool Hand Luke",
"released": 1967,
"color": true,
"Actors": [
"Paul Newman"
]
},
{
"Title": "Bullitt",
"released": 1968,
"color": true,
"Actors": [
"Steve McQueen",
"Jacqueline Bisset"
]
}
]
反序列化
反序列化使用Unmarshal
函数:
func Unmarshal(data []byte, v interface{}) error
1. 明确知道 JSON 格式
我们要先将 JSON 格式表示为一个确定的类型。
1.如下的 JSON 格式:
{ "name": "Awesome 4K", "resolutions": [ { "width": 1280, "height": 720 }, { "width": 1920, "height": 1080 }, { "width": 3840, "height": 2160 } ] }
我们可以用如下结构体来表示它:
struct { Name string Resolutions []struct { Width int Height int } }
2.如下的 JSON 格式:
{ "height": 720, "width": 1280 }
也可以使用map[string]int
,也就是字典来表示。
3.如下的 JSON 格式:
[ { "width": 1280, "height": 720 }, { "width": 1920, "height": 1080 }, { "width": 3840, "height": 2160 } ]
使用切片[]map[string]int
来表示。
不管怎样,一个确定的JSON格式总是可以使用切片、结构体或字典来表示。
之后就可以执行反序列化了:
var jsonBlob = []byte(` [ { "width": 1280, "height": 720 }, { "width": 1920, "height": 1080, }, { "width": 3840, "height": 2160 } ] `) di := []map[string]int{} err = json.Unmarshal(jsonBlob, &di) if err != nil { fmt.Println("error:", err) } fmt.Printf("%+v\n", di)
输出
[map[height:720 width:1280] map[height:1080 width:1920] map[height:2160 width:3840]]
2. 无法确定 JSON 格式
无法确定的格式可以直接使用interface{}
类型来表示。这个时候,如果是JSON对象,则反序列化时 json 库会使用map[string]interface{}
类型来表示它。如果是JSON数组,则会使用[]interface{}
类型来表示它。具体interface{}
对应的具体类型,就需要使用类型断言了。
具体看一个示例:
package main import ( "encoding/json" "fmt" ) func main() { var jsonBlob = []byte(` { "name": "Awesome 4K", "price": 1999.9, "resolutions": [ { "width": 1280, "height": 720 }, { "width": 1920, "height": 1080 }, { "width": 3840, "height": 2160 } ] } `) var d interface{} err := json.Unmarshal(jsonBlob, &d) if err != nil { fmt.Println("error:", err) } fmt.Println(d) m := d.(map[string]interface{}) for k, v := range m { switch vv := v.(type) { case string: fmt.Println(k, "is string", vv) case float64: fmt.Println(k, "is float64", vv) case []interface{}: fmt.Println(k, "is an array:") for i, u := range vv { fmt.Println(i, u) } default: fmt.Println(k, "is of a type I don't know how to handle") } } }
输出:
map[name:Awesome 4K price:1999.9 resolutions:[map[height:720 width:1280] map[height:1080 width:1920] map[height:2160 width:3840]]]
resolutions is an array:
0 map[height:720 width:1280]
1 map[height:1080 width:1920]
2 map[height:2160 width:3840]
name is string Awesome 4K
price is float64 1999.9
终于介绍完啦!小伙伴们,这篇关于《Go 语言 JSON 标准库的使用》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!
-
369 收藏
-
344 收藏
-
329 收藏
-
377 收藏
-
141 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习