当前位置:首页 > 文章列表 > Golang > Go教程 > 深入解析Golang中JSON的编码与解码

深入解析Golang中JSON的编码与解码

来源:脚本之家 2023-05-12 18:37:16 0浏览 收藏

你在学习Golang相关的知识吗?本文《深入解析Golang中JSON的编码与解码》,主要介绍的内容就涉及到JSON、解码、编码,如果你想提升自己的开发能力,就不要错过这篇文章,大家要知道编程理论基础和实战操作都是不可或缺的哦!

随着互联网的快速发展和数据交换的广泛应用,各种数据格式的处理成为软件开发中的关键问题。JSON 作为一种通用的数据交换格式,在各种应用场景中都得到了广泛应用,包括 Web 服务、移动应用程序和大规模数据处理等。Golang 作为一种开发高性能、并发安全的语言,具备出色的处理 JSON 的能力。本文将介绍 Golang 中 JSON 编码与解码的相关知识,帮助大家了解其基本原理和高效应用。

1. JSON 简介

JSON 是一种基于文本的轻量级数据交换格式,它以易于人类阅读和编写的方式表示结构化数据。JSON 采用键值对的形式组织数据,支持多种数据类型,包括字符串、数字、布尔值、数组和对象等。以下是一个简单的 JSON 示例:

 {
   "name": "Alice",
   "age": 25,
   "isStudent": true,
   "hobbies": ["reading", "coding", "music"]
 }

在上述示例中,name 是一个字符串类型的键,对应的值是 "Alice";age 是一个数字类型的键,对应的值是 25;isStudent 是一个布尔类型的键,对应的值是 true;hobbies 是一个数组类型的键,对应的值是一个包含三个字符串元素的数组。

2. Golang 中的 JSON 编码

Golang 标准库中的 encoding/json 包提供了丰富的功能,用于将 Go 数据结构编码为 JSON 格式。下面是一些常见的 JSON 编码用法示例:

2.1 结构体的 JSON 编码

在 Golang 中,可以通过给结构体字段添加 json 标签来指定 JSON 编码时的字段名和其他选项。例如,考虑以下 Person 结构体:

 type Person struct {
     Name string `json:"name"`
     Age  int    `json:"age"`
 }

要将该结构体编码为 JSON,可以使用 json.Marshal() 函数:

 p := Person{Name: "Alice", Age: 25}
 data, err := json.Marshal(p)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(string(data))

运行上述代码,输出结果将是:

{"name":"Alice","age":25}

在这个例子中,json.Marshal() 函数将 Person 结构体编码为 JSON 格式,并将结果存储在 data 变量中。最后,我们使用 fmt.Println() 函数将编码后的 JSON 字符串打印出来。

2.2 切片和映射的 JSON 编码

除了结构体,Golang 中的切片和映射也可以方便地进行 JSON 编码。例如,考虑以下切片和映射的示例:

 names := []string{"Alice", "Bob", "Charlie"}
 data, err := json.Marshal(names)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(string(data))
 ​
 scores := map[string]int{
     "Alice":   100,
     "Bob":     85,
     "Charlie": 92,
 }
 data, err = json.Marshal(scores)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(string(data))

运行上述代码,输出结果将是:

 ["Alice","Bob","Charlie"]
 {"Alice":100,"Bob":85,"Charlie":92}

在这个例子中,我们首先将切片 names 和映射 scores 分别进行 JSON 编码,并将结果打印出来。切片和映射会被编码为对应的 JSON 数组和对象。

3. Golang 中的 JSON 解码

除了 JSON 编码,Golang 中的 encoding/json 包还提供了 JSON 解码的功能,可以将 JSON 数据解码为 Go 数据结构。下面是一些常见的 JSON 解码用法示例:

3.1 JSON 解码为结构体

要将 JSON 解码为结构体,需要先定义对应的结构体类型,并使用 json.Unmarshal() 函数进行解码。例如,考虑以下 JSON 数据:

 {
   "name": "Alice",
   "age": 25
 }

我们可以定义一个 Person 结构体来表示这个数据:

 type Person struct {
     Name string `json:"name"`
     Age  int    `json:"age"`
 }

然后,可以使用 json.Unmarshal() 函数将 JSON 解码为该结构体:

 jsonStr := `{"name":"Alice","age":25}`
 var p Person
 err := json.Unmarshal([]byte(jsonStr), &p)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(p.Name, p.Age)

运行上述代码,输出结果将是:

Alice 25

在这个例子中,我们首先将 JSON 数据保存在 jsonStr 变量中。然后,使用 json.Unmarshal() 函数将 JSON 解码为 Person 结构体,并将结果存储在变量 p 中。最后,我们打印出 p 的字段值。

3.2 JSON 解码为切片和映射

除了解码为结构体,JSON 数据还可以解码为切片和映射。解码为切片和映射的过程与解码为结构体类似。以下是示例代码:

 jsonStr := `["Alice","Bob","Charlie"]`
 var names []string
 err := json.Unmarshal([]byte(jsonStr), &names)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(names)
 ​
 jsonStr = `{"Alice":100,"Bob":85,"Charlie":92}`
 var scores map[string]int
 err = json.Unmarshal([]byte(jsonStr), &scores)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(scores)

运行上述代码,输出结果将是:

 [Alice Bob Charlie]
 map[Alice:100 Bob:85 Charlie:92]

在这个例子中,我们首先将 JSON 数据保存在 jsonStr 变量中。然后,使用 json.Unmarshal() 函数将 JSON 解码为相应的切片和映射,并将结果存储在对应的变量中。最后,我们打印出这些变量的值。

4. 自定义编码与解码

Golang 的 encoding/json 包提供了一种自定义编码与解码的方式,可以灵活地控制 JSON 数据的序列化和反序列化过程。通过实现 json.Marshaler 和 json.Unmarshaler 接口,可以定制字段的编码和解码行为。

例如,假设我们有一个时间类型的字段,我们希望在 JSON 中以特定的日期格式进行编码和解码。我们可以定义一个自定义类型,并实现 json.Marshaler 和 json.Unmarshaler 接口。

 type CustomTime time.Time
 ​
 func (ct CustomTime) MarshalJSON() ([]byte, error) {
     formatted := time.Time(ct).Format("2006-01-02")
     return []byte(`"` + formatted + `"`), nil
 }
 ​
 func (ct *CustomTime) UnmarshalJSON(data []byte) error {
     // 假设日期格式为 "2006-01-02"
     parsed, err := time.Parse(`"2006-01-02"`, string(data))
     if err != nil {
         return err
     }
     *ct = CustomTime(parsed)
     return nil
 }

在上述代码中,我们定义了一个 CustomTime 类型,并为它实现了 MarshalJSON() 和 UnmarshalJSON() 方法。MarshalJSON() 方法将时间格式化为指定的日期格式,并进行编码。UnmarshalJSON() 方法根据特定的日期格式解码 JSON 数据并转换为时间类型。

通过自定义编码和解码逻辑,我们可以根据实际需求灵活处理特定类型的字段。

5. JSON 标签选项

除了指定字段名,json 标签还提供了其他选项,以进一步控制编码和解码的行为。以下是一些常用的 JSON 标签选项:

  • omitempty:如果字段的值为空值(如零值、空字符串、空切片等),则在编码时忽略该字段。
  • string:将字段编码为JSON字符串类型,而不是其原始类型。
  • omitempty 和 string 可以组合使用,例如 json:"myField,omitempty,string"。

示例:

 type Person struct {
     Name      string    `json:"name"`
     Age       int       `json:"age,omitempty"`
     BirthDate CustomTime `json:"birth_date,string"`
 }

在上述示例中,我们定义了一个 Person 结构体,其中 Name 字段的编码和解码使用默认选项,Age 字段使用 omitempty 选项,BirthDate 字段使用 string 选项。

这些选项可以帮助我们更精确地控制 JSON 数据的编码和解码过程。

6. 处理嵌套结构体

在处理复杂的数据结构时,结构体可能会嵌套其他结构体。Golang 的 JSON 编码与解码能够自动处理嵌套结构体,无需额外的配置。

例如,假设我们有以下是关于处理嵌套结构体的示例代码:

 type Address struct {
     Street  string `json:"street"`
     City    string `json:"city"`
     Country string `json:"country"`
 }
 ​
 type Person struct {
     Name    string  `json:"name"`
     Age     int     `json:"age"`
     Address Address `json:"address"`
 }
 ​
 p := Person{
     Name: "Alice",
     Age:  25,
     Address: Address{
         Street:  "123 Main St",
         City:    "New York",
         Country: "USA",
     },
 }
 ​
 data, err := json.Marshal(p)
 if err != nil {
     log.Fatal(err)
 }
 fmt.Println(string(data))

在上述代码中,我们定义了两个结构体:Address 和 Person。Person 结构体中嵌套了 Address 结构体作为其中一个字段。

我们创建了一个 Person 的实例 p,并将其编码为 JSON 格式。json.Marshal() 函数会自动递归地将嵌套结构体编码为嵌套的 JSON 对象。

输出结果将是:

 {"name":"Alice","age":25,"address":{"street":"123 Main St","city":"New York","country":"USA"}}

通过 Golang 的 JSON 编码与解码功能,我们可以轻松处理具有嵌套结构的复杂数据。

7. 处理非导出字段

在 Golang 中,非导出(未以大写字母开头)的结构体字段默认在 JSON 编码和解码过程中会被忽略。这意味着这些字段不会被编码到 JSON 中,也不会从 JSON 中解码。

如果需要处理非导出字段,可以在字段的定义中使用 json:"-" 标签,表示忽略该字段。或者,可以通过定义自定义的 MarshalJSON 和 UnmarshalJSON 方法来处理非导出字段的编码和解码逻辑。

 type Person struct {
     name string `json:"-"`
     Age  int    `json:"age"`
 }

在上述示例中,name 字段被标记为忽略,不会参与 JSON 编码与解码。Age 字段会被正常编码和解码。

8. 处理空值

在 JSON 编码与解码过程中,空值的处理是一个重要的考虑因素。空值包括nil指针、空切片、空映射等。Golang 的 encoding/json 包提供了对空值的处理选项。

在编码时,如果字段的值是空值,可以使用 omitempty 选项指示在编码时忽略该字段。这对于减少 JSON 数据中的冗余信息很有用。

在解码时,如果 JSON 数据中的字段的值是 null,可以使用指针类型或 interface{} 类型来接收解码后的值。这样可以区分出空值和非空值。

示例:

 type Person struct {
     Name  string  `json:"name,omitempty"`
     Age   int     `json:"age,omitempty"`
     Extra *string `json:"extra,omitempty"`
 }
 ​
 jsonStr := `{"name":"Alice","age":null,"extra":"additional info"}`
 var p Person
 err := json.Unmarshal([]byte(jsonStr), &p)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(p.Name)   // 输出: "Alice"
 fmt.Println(p.Age)    // 输出: 0
 fmt.Println(p.Extra)  // 输出: nil

在上述示例中,Person 结构体中的 Name 字段使用了 omitempty 选项,因此在编码时如果字段的值为空字符串,则会被忽略。Age 字段在 JSON 数据中的值为 null,解码后会被设置为类型的零值。Extra 字段在 JSON 数据中的值为 "additional info",解码后被设置为 nil。

9. 处理循环引用

循环引用是指一个数据结构中的对象相互引用,形成了闭环。在进行 JSON 编码与解码时,处理循环引用是一个挑战。

Golang 的 encoding/json 包默认不支持循环引用的编码与解码,因为会导致无限递归。如果存在循环引用的数据结构,需要额外的处理来避免循环引用。

一种处理循环引用的方法是使用指针类型来打破循环。通过将结构体字段定义为指针类型,可以在 JSON 编码与解码过程中避免循环引用。

示例:

 type Person struct {
     Name    string   `json:"name"`
     Friends []*Person `json:"friends"`
 }
 ​
 alice := &Person{Name: "Alice"}
 bob := &Person{Name: "Bob"}
 charlie := &Person{Name: "Charlie"}
 ​
 alice.Friends = []*Person{bob, charlie}
 bob.Friends = []*Person{alice}
 charlie.Friends = []*Person{alice, bob}
 ​
 data, err := json.Marshal(alice)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(string(data))

在上述示例中,Person 结构体中的 Friends 字段被定义为 []*Person 在进行 JSON 编码时,Golang 的 encoding/json 包会处理循环引用,并将循环引用中的对象替换为null。

在解码 JSON 数据时,Golang 的 encoding/json 包默认情况下无法处理循环引用。如果 JSON 数据中存在循环引用,解码过程将会进入无限递归,并最终导致堆栈溢出。为了解决这个问题,我们可以使用 json.RawMessage 类型或自定义解码函数来处理循环引用。

使用 json.RawMessage 类型,可以在结构体中存储原始的 JSON 数据,然后在后续的处理中进行解析。

示例:

 type Person struct {
     Name    string          `json:"name"`
     Friends []json.RawMessage `json:"friends"`
 }
 ​
 jsonStr := `{"name":"Alice","friends":[
     {"name":"Bob","friends":null},
     {"name":"Charlie","friends":null}
 ]}`
 var p Person
 err := json.Unmarshal([]byte(jsonStr), &p)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(p.Name)     // 输出: "Alice"
 fmt.Println(p.Friends)  // 输出: [{"name":"Bob","friends":null},{"name":"Charlie","friends":null}]

在上述示例中,Person 结构体中的 Friends 字段使用了 json.RawMessage 类型,它会将原始的 JSON 数据存储为字节切片。这样,我们可以在后续的处理中解析这些原始数据。

自定义解码函数是另一种处理循环引用的方法。通过自定义解码函数,我们可以控制解码过程,处理循环引用并构建正确的对象关系。

示例:

 type Person struct {
     Name    string   `json:"name"`
     Friends []*Person `json:"friends"`
 }
 ​
 func (p *Person) UnmarshalJSON(data []byte) error {
     type Alias Person
     aux := &struct {
         *Alias
         Friends []*Person `json:"friends"`
     }{
         Alias: (*Alias)(p),
     }
     if err := json.Unmarshal(data, &aux); err != nil {
         return err
     }
     p.Friends = aux.Friends
     return nil
 }
 ​
 jsonStr := `{"name":"Alice","friends":[
     {"name":"Bob","friends":null},
     {"name":"Charlie","friends":null}
 ]}`
 var p Person
 err := json.Unmarshal([]byte(jsonStr), &p)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(p.Name)            // 输出: "Alice"
 fmt.Println(p.Friends[0].Name) // 输出: "Bob"
 fmt.Println(p.Friends[1].Name) // 输出: "Charlie"

在上述示例中,我们为 Person 结构体定义了自定义的解码函数 UnmarshalJSON。在解码过程中,我们使用一个辅助结构体 aux 来接收解码的 JSON 数据,并将其转换为 Person 结构体。然后,将辅助结构体中的 Friends 字段赋值给原始结构体的 Friends 字段。

通过使用 json.RawMessage 类型或自定义解码函数,我们可以处理包含循环引用的 JSON 数据,并成功地解码成正确的对象结构。

10. 处理不确定结构的 JSON 数据

有时,我们可能需要处理具有不确定结构的 JSON 数据。这种情况下,Golang 的 encoding/json 包提供了 json.RawMessage 类型和 interface{} 类型来处理这种不确定性。

json.RawMessage 类型可以用于存储原始的 JSON 数据,并在后续的处理中解析。它可以接收任何合法的 JSON 数据,并保留其原始形式。

示例:

 type Data struct {
     Name    string          `json:"name"`
     Payload json.RawMessage `json:"payload"`
 }
 ​
 jsonStr := `{"name":"Event","payload":{"type":"message","content":"Hello, world!"}}`
 var d Data
 err := json.Unmarshal([]byte(jsonStr), &d)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(d.Name)                   // 输出: "Event"
 fmt.Println(string(d.Payload))        // 输出: {"type":"message","content":"Hello, world!"}

在上述示例中,Data 结构体中的 Payload 字段使用了 json.RawMessage 类型,它会将原始的 JSON 数据存储为字节切片。我们可以使用 string() 函数将其转换为字符串进行打印或进一步解析。

另一种处理不确定结构的方法是使用 interface{} 类型。interface{} 类型可以接收任何类型的值,包括基本类型、结构体、切片等。通过使用 interface{} 类型,我们可以处理具有不确定结构的 JSON 数据,但在后续的处理中需要进行类型断言。

示例:

 type Data struct {
     Name    string      `json:"name"`
     Payload interface{} `json:"payload"`
 }
 ​
 jsonStr := `{"name":"Event","payload":{"type":"message","content":"Hello, world!"}}`
 var d Data
 err := json.Unmarshal([]byte(jsonStr), &d)
 if err != nil {
     log.Fatal(err)
 }
 ​
 fmt.Println(d.Name)                                // 输出: "Event"
 payload, ok := d.Payload.(map[string]interface{})
 if ok {
     fmt.Println(payload["type"].(string))          // 输出: "message"
     fmt.Println(payload["content"].(string))       // 输出: "Hello, world!"
 }

在上述示例中,Data 结构体中的Payload字段使用了 interface{} 类型,它可以接收任何类型的值。在后续的处理中,我们使用类型断言将其转换为具体的类型,并进行进一步的操作。

通过使用 json.RawMessage 类型和 interface{} 类型,我们可以灵活地处理不确定结构的 JSON 数据,并根据实际情况进行解析和操作。

11. 总结

本文深入介绍了 Golang 中的 JSON 编码与解码技术。我们了解了 JSON 的基本原理和 Golang 中处理 JSON 的方法。通过示例代码,我们展示了如何使用 encoding/json 包进行编码和解码操作,并通过合理应用这些技术,我们可以高效处理大规模的结构化数据,提高软件的性能和效率。

本篇关于《深入解析Golang中JSON的编码与解码》的介绍就到此结束啦,但是学无止境,想要了解学习更多关于Golang的相关知识,请关注golang学习网公众号!

版本声明
本文转载于:脚本之家 如有侵犯,请联系study_golang@163.com删除
Golang中间件设计示例详解Golang中间件设计示例详解
上一篇
Golang中间件设计示例详解
深入探究Go语言从反射到元编程的实践与探讨
下一篇
深入探究Go语言从反射到元编程的实践与探讨
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    542次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    508次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    497次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    484次学习
查看更多
AI推荐
  • 笔灵AI生成答辩PPT:高效制作学术与职场PPT的利器
    笔灵AI生成答辩PPT
    探索笔灵AI生成答辩PPT的强大功能,快速制作高质量答辩PPT。精准内容提取、多样模板匹配、数据可视化、配套自述稿生成,让您的学术和职场展示更加专业与高效。
    9次使用
  • 知网AIGC检测服务系统:精准识别学术文本中的AI生成内容
    知网AIGC检测服务系统
    知网AIGC检测服务系统,专注于检测学术文本中的疑似AI生成内容。依托知网海量高质量文献资源,结合先进的“知识增强AIGC检测技术”,系统能够从语言模式和语义逻辑两方面精准识别AI生成内容,适用于学术研究、教育和企业领域,确保文本的真实性和原创性。
    20次使用
  • AIGC检测服务:AIbiye助力确保论文原创性
    AIGC检测-Aibiye
    AIbiye官网推出的AIGC检测服务,专注于检测ChatGPT、Gemini、Claude等AIGC工具生成的文本,帮助用户确保论文的原创性和学术规范。支持txt和doc(x)格式,检测范围为论文正文,提供高准确性和便捷的用户体验。
    28次使用
  • 易笔AI论文平台:快速生成高质量学术论文的利器
    易笔AI论文
    易笔AI论文平台提供自动写作、格式校对、查重检测等功能,支持多种学术领域的论文生成。价格优惠,界面友好,操作简便,适用于学术研究者、学生及论文辅导机构。
    37次使用
  • 笔启AI论文写作平台:多类型论文生成与多语言支持
    笔启AI论文写作平台
    笔启AI论文写作平台提供多类型论文生成服务,支持多语言写作,满足学术研究者、学生和职场人士的需求。平台采用AI 4.0版本,确保论文质量和原创性,并提供查重保障和隐私保护。
    34次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码