脚本之家,脚本语言编程技术及教程分享平台!
分类导航

Python|VBS|Ruby|Lua|perl|VBA|Golang|PowerShell|Erlang|autoit|Dos|bat|

服务器之家 - 脚本之家 - Golang - golang中json操作的完全指南

golang中json操作的完全指南

2022-07-14 13:57grb123 Golang

golang标准库有一个名叫encoding/json的包,包含了JSON的序列化(Marshal)和反序列化(Unmarshal)的能力,下面这篇文章主要给大家介绍了关于golang中json操作的相关资料,需要的朋友可以参考下

前言

JSON是一种轻量级的数据交换格式。易于阅读和编写。 golang 提供了 encoding/json 包来操作JSON数据。

1. 结构体与JSON互转

(1)使用 json.Marshal() 方法,把结构体转成 JSON字符串

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
import (
    "encoding/json"
    "fmt"
)
 
type Student struct {
    Name string
    Age int
    Skill string
}
 
func main()  {
    stu := Student{"tom", 12, "football"}
    data, err := json.Marshal(&stu)
    if err != nil {
        fmt.Printf("序列化错误 err=%v\n", err)
        return
    }
    fmt.Println("序列化后: ", string(data))
}

打印: 序列化后: {“Name”:“tom”,“Age”:12,“Skill”:“football”}

(2)JSON字符串 转 结构体,可以使用 json.Unmarshal()方法

?
1
2
3
4
5
6
7
8
9
10
func main()  {
    str := `{"Name":"tom","Age":12,"Skill":"football"}`
    var stu2 Student
    err := json.Unmarshal([]byte(str), &stu2)
    if err != nil {
        fmt.Printf("反序列化错误 err=%v\n", err)
        return
    }
    fmt.Printf("反序列化后: Student=%v, Name=%v\n", stu2, stu2.Name)
}

打印: 反序列化后: Student={tom 12 football}, Name=tom

(3)如何实现结构体序列化后key的名称能自定义

对于自定义key的名称,可以给 struct变量指定一个tag标签

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
type Student struct {
    Name string   `json:"stu_name"`
    Age int       `json:"stu_age"`
    Skill string  // 也可以不指定 tag标签,默认就是 变量名称
}
 
func main()  {
    stu := Student{"tom", 12, "football"}
    data, err := json.Marshal(&stu)
    if err != nil {
        fmt.Printf("序列化错误 err=%v\n", err)
        return
    }
    fmt.Println("序列化后: ", string(data))
}

打印后,可以看到 key的名称已经变成了我们指定的 tag标签 的名称

序列化后: {“stu_name”:“tom”,“stu_age”:12,“Skill”:“football”}

2. map与JSON互转

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
func main()  {
    // map 转 Json字符串
    m := make(map[string]interface{})
    m["name"] = "jetty"
    m["age"] = 16
 
    data, err := json.Marshal(&m)
    if err != nil {
        fmt.Printf("序列化错误 err=%v\n", err)
        return
    }
    fmt.Println("序列化后: ", string(data))    // 打印: 序列化后:  {"age":16,"name":"jetty"}
 
    // Json字符串 转 map
    str := `{"age":25,"name":"car"}`
    err = json.Unmarshal([]byte(str), &m)
    if err != nil {
        fmt.Printf("反序列化错误 err=%v\n", err)
        return
    }
    fmt.Printf("反序列化后: map=%v, name=%v\n", m, m["name"])
    // 打印: 反序列化后: map=map[age:25 name:car], name=car
}

3. 结构体的变量不加tag标签能否正常转成json数据

如果变量首字母小写,则为private。因为取不到反射信息,不能转。

如果变量首字母大写,则为public。不管加不加 tag 都能正常转,加了tag的变量就按照tag的名称显示。

示例:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
type User struct {
    Name string    `json:"u_name"`
    age int        `json:"u_age"`
    Skill string   // 也可以不指定 tag标签,默认就是 变量名称
    addr string
}
 
func main()  {
    user := User{"admin", 23, "football", "上海"}
    data, err := json.Marshal(&user)
    if err != nil {
        fmt.Printf("序列化错误 err=%v\n", err)
        return
    }
    fmt.Println("序列化后: ", string(data))  // 打印: 序列化后:  {"u_name":"admin","Skill":"football"}
}

通过打印,我们发现小写的变量,如 age、addr 都没有转成 json数据。

总结:

首字母小写的不管加不加tag都不能转为json数据,而大写的加了tag可以取别名,不加tag则json内的字段跟结构体变量原名一致

4. JSON操作的一些小技巧

(1)忽略掉 struct 指定字段

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
type User struct {
    Name string    `json:"u_name"`
    Password string `json:"password"`
    Email string `json:"email"`
}
 
func main()  {
    user := User{"admin", "pwd", "user@163.com"}
    person := Person{23, "上海"}
    // 忽略掉 Password 字段
    data, _ := json.Marshal(struct {
        *User
        Password string `json:"password,omitempty"`
    }{User: &user})
    fmt.Println("忽略字段: ", string(data))  // 打印: 忽略字段: {"u_name":"admin","email":"user@163.com"}
}

忽略字段: {“u_name”:“admin”,“email”:“user@163.com”}}

(2)添加额外的字段

?
1
2
3
4
5
6
7
8
9
data, _ = json.Marshal(struct {
    *User
    Skill string `json:"skill"// 临时添加额外的 Skill字段
}{
    User: &user,
    Skill: "football",
})
 
fmt.Println("添加额外字段: ", string(data))

添加额外字段: {“u_name”:“admin”,“password”:“pwd”,“email”:“user@163.com”,“skill”:“football”}

(3)合并两个 struct

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
type User struct {
    Name string    `json:"u_name"`
    Password string `json:"password"`
    Email string `json:"email"`
}
 
type Person struct {
    Age int
    Addr string `json:"addr"`
}
 
func main()  {
    // 初始化两个 struct
    user := User{"admin", "pwd", "user@163.com"}
    person := Person{23, "上海"}
    
    data, _ := json.Marshal(struct {
        *User
        *Person
    }{
        User: &user,
        Person: &person,
    })
    
    fmt.Println("合并两个struct: ", string(data))
}

合并两个struct: {“u_name”:“admin”,“password”:“pwd”,“email”:“user@163.com”,“Age”:23,“addr”:“上海”}

(4)字符串传递给 int类型

?
1
2
3
4
5
6
7
8
9
emp := struct {                    // 创建匿名 struct
    Num int `json:"num,string"`
}{15,}
data, _ := json.Marshal(&emp)
fmt.Println("数字转成字符串: ", string(data))       // 数字转成字符串: {"num":"15"}
 
str := `{"Num":"25"}`
_ = json.Unmarshal([]byte(str), &emp)
fmt.Printf("字符串转成数字: Emp.Num=%v\n", emp.Num) // 字符串转成数字: Emp.Num=25

(5)一个 json 分成两个struct

?
1
2
3
4
5
6
7
8
9
10
11
12
str = ` {"u_name":"system","password":"abc","email":"user2@163.com","Age":23,"addr":"杭州"}`
var user2 User
var person2 Person
_ := json.Unmarshal([]byte(str), &struct {
    *User
    *Person
}{
    User: &user2,
    Person: &person2,
})
 
fmt.Printf("分成两个struct: User=%v, Person=%v\n", user2, person2)

分成两个struct: User={system abc user2@163.com}, Person={23 杭州}

补充:GoLang json格式化输出

简单记录一下go语言json格式化输出的办法

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
import (
    "bytes"
    "encoding/json"
    "fmt"
    "os"
)
 
type Complex_Type struct{
    Age int `json:"age"`
    Name string `json:"name"`
    Grades map[string]int `json:"grade"`
    Parents []string `json:"parents"`
}
 
    grades := map[string]int{
        "math" : 96,
        "chinese" : 87,
        "english" : 93,
    }
 
    parents := []string{
        "minato",
        "kushina",
    }
 
    complex_type := Complex_Type{
        Age: 18,
        Name: "Naruto",
        Grades : grades,
        Parents: parents,
    }
    res,err := json.Marshal(complex_type)
    exitOnError(err)
    var out bytes.Buffer
    err = json.Indent(&out,res,"","\t")
    out.WriteTo(os.Stdout)
    fmt.Printf("\n")

输出:

golang中json操作的完全指南

总结

到此这篇关于golang中json操作的文章就介绍到这了,更多相关golang json操作内容请搜索服务器之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持服务器之家!

原文链接:https://blog.csdn.net/grb123/article/details/123017658

延伸 · 阅读

精彩推荐
  • GolangGo中map数据类型3点小知识

    Go中map数据类型3点小知识

    这篇文章主要介绍了Go中map数据类型3点小知识,在go中使用map时可能会用本文中的小知识,需要的朋友可以参考下 ...

    junjie3602019-11-28
  • GolangGo语言正则表达式的使用详解

    Go语言正则表达式的使用详解

    正则表达式是一种进行模式匹配和文本操纵的功能强大的工具。这篇文章主要介绍了Go正则表达式使用,本文给大家介绍的非常详细,对大家的工作或学习具...

    大漠胡萝卜3072020-06-06
  • GolangGo 语言的函数与指针

    Go 语言的函数与指针

    函数就是一段基本的代码块,一般用来对需要重复执行的代码进行复用。在 go 中,函数是『一等公民』,这与 js 类似,也就是可以将函数当做一个变量进...

    自然醒的笔记本11042021-04-14
  • Golangxorm根据数据库生成go model文件的操作

    xorm根据数据库生成go model文件的操作

    这篇文章主要介绍了xorm根据数据库生成go model文件的操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    zgcbj10142021-03-08
  • Golanggolang之tcp自动重连实现方法

    golang之tcp自动重连实现方法

    下面小编就为大家带来一篇golang之tcp自动重连实现方法。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧 ...

    jingxian8082020-05-07
  • GolangGolang 如何判断数组某个元素是否存在 (isset)

    Golang 如何判断数组某个元素是否存在 (isset)

    这篇文章主要介绍了Golang 如何判断数组某个元素是否存在 (isset),具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧...

    潘广宇 Leo19572021-06-07
  • GolangGo语言中切片使用的注意事项小结

    Go语言中切片使用的注意事项小结

    切片是引用类型,相信对大家来说都不陌生,下面这篇文章主要给大家总结介绍了关于Go语言中切片使用的一些注意事项,文中通过示例代码介绍的非常详...

    Aaaaaaaaaaayou3602020-05-13
  • GolangGo语言字符串高效拼接的实现

    Go语言字符串高效拼接的实现

    这篇文章主要介绍了Go语言字符串高效拼接的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下...

    志强12242412020-05-22