Go语言——Json处理

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
全局流量管理 GTM,标准版 1个月
云解析 DNS,旗舰版 1个月
简介: Go语言——Json处理

□ JSON处理

JSON (JavaScript Object Notation)是一种比XML更轻量级的数据交换格式,在易于人们阅读和编写的同时,也易于程序解析和生成。

 尽管JSON是JavaScript的一个子集,但JSON采用完全独立于编程语言的文本格式,且表现为键/值对集合的文本描述形式(类似一些编程语言中的字典结构),这使它成为较为理想的、跨平台、跨语言的数据交换语言。

 开发者可以用 JSON 传输简单的字符串、数字、布尔值,也可以传输一个数组,或者一个更复杂的复合结构。

 在 Web 开发领域中, JSON被广泛应用于 Web 服务端程序和客户端之间的数据通信。

 Go语言内建对JSON的支持。使用Go语言内置的encoding/json 标准库,开发者可以轻松使用Go程序生成和解析JSON格式的数据。

JSON官方网站:http://www.json.org/

在线格式化:http://www.json.cn/

1. 编码JSON (编组|封送 Marshaling)

1. Marhsal()

将数据编码成json字符串

// func Marshal(v interface{}) ([]byte, error){}
// 示例代码:
// Test11 go json.Marshal时,结构体字段需要大写 否则Marshal时不显示
type Test11 struct {
  Name string
  age  string
}
func main() {
  t1 := Test11{"sb", "12"}
  //生成一段JSON格式的文本
  //如果编码成功, err 将赋于零值 nil,变量b 将会是一个进行JSON格式化之后的[]byte类型
  b, err := json.Marshal(t1)
  if err != nil {
    fmt.Println(err)
  } else {
    fmt.Println(b)         // [123 34 78 97...]
    fmt.Println(string(b)) // {"Name":"sb"}
  }
}

2. MarshalIndent

将数据编码成json字符串 例如map stuct等

// MarshalIndent 很像 Marshal,只是用缩进对输出进行格式化
func MarshalIndent(v interface{}, prefix, indent string) ([]byte, error)
// v: 想要编码成json的结构体|map
// prefix: 前缀,通常就设置为""即可。
// indent:缩进,通常设置为制表符\t即可
//示例代码:
func main() {
  t1 := Test11{"sb", 12}
  b,_ := json.MarshalIndent(t1,"","\t")
  fmt.Println(string(b))
  // {
  //   "Name": "sb",
  //   "Age": 12
  // }
}

MarshalIndent相比于Marhsal()的区别就在于读结果做了Indent 的处理:

Indent 的代码有点长,简单说就是对Json 多了一些格式处理。

3. struct tag

我们看到上面的输出字段名的首字母都是大写的,如果你想用小写的首字母怎么办呢?把结构体的字段名改成首字母小写的?

JSON输出的时候必须注意,只有导出的字段(首字母是大写)才会被输出,如果修改字段名,那么就会发现什么都不会输出,所以必须通过struct tag定义来实现。

针对JSON的输出,我们在定义struct tag的时候需要注意的几点是:

字段的tag是"-",那么这个字段不会输出到JSON

tag中带有自定义名称,那么这个自定义名称会出现在JSON的字段名中

tag中如果带有"omitempty"选项,那么如果该字段值为空,就不会输出到JSON串中

如果字段类型是bool, string, int, int64等,而tag中带有",string"选项,那么这个字段在输出到JSON的时候会把该字段对应的值转换成JSON字符串

实例代码:

type Test11 struct {
  // 1. 字段的tag是"-",那么这个字段不会输出到JSON
  Name string `json:"-"`
  // 2. 转换为字符串,再输出
  Age  int `json:",string"` // --- "Age": "12",
    //  Age  int `json:"Age"` // --- "Age": 12,
  // 3. Subjects 的值会进行二次JSON编码
  Sex string `json:"Sex"`
  // 4. 如果 Height 为空,则不输出到JSON串中 没有这个tag 会显示 Height:""
  Height string `json:"Height,omitempty"`
  // 5. tag中带有自定义名称,那么这个自定义名称会出现在JSON的字段名中
  Score int `json:"成绩"`
}
func main() {
  t1 := Test11{Name: "sb", Age:12,Sex:"男",Score: 98}
  b,_ := json.MarshalIndent(t1,"","\t")
  fmt.Println(string(b))
}

4. 通过map生成JSON

func main() {
  m:= map[string]string{"one":"你好","two":"我是","three":"汪汪队"}
    v,err :=json.MarshalIndent(m,"","\t")
    if err!=nil{
        fmt.Println(err)      //错误信息
    }else{
        fmt.Println(string(v))
    }
    // {
    //   "one": "你好",
    //   "three": "汪汪队",
    //   "two": "我是"
    // }
}

2. 解码JSON(解组|解封送:Unmarshaling)

Json Unmarshal:将json字符串解码到相应的数据结构。

func Unmarshal(data []byte, v interface{}) error
// data:想要json解码的[]byte类型数据
// v:将data Json解码到v(可以为结构体|map)里
// 实例代码
func main() {
  t1 := Test11{Name: "sb", Age:12,Sex:"男",Score: 98}
  val,_ := json.MarshalIndent(t1,"","\t")
  fmt.Println(string(val)) // 上回的json
    // {
    //   "Age": "12",
    //   "Sex": "男",
    //   "成绩": 98
    // }
  var a interface{}
  err := json.Unmarshal(val,&a)
  if err!=nil {
    fmt.Println(err)
  }
  fmt.Println(a) // map[Age:12 Sex:男 成绩:98]
  fmt.Printf("%T\n",a) // map[string]interface {}
}

Unmarshal (解组器) 解析JSON编码的数据并存储结果到 v指向的值中。

如果v为nil或不是指针,Unmarshal 返回InvalidUnmarshalError。

要将JSON解组到结构中,Unmarshal (解组) 匹配传入对象 Marshaling(编组) 处理使用的键的键(结构字段名或其标记),更喜欢精确匹配,但也接受不区分大小写的匹配。

通过默认情况下,没有相应结构字段的对象键是已忽略(也就是说你有的字段,就解析给你(如字段Age);你没有的字段,就忽略(如字段Sex);它没有的字段,你就为默认值(如字段S))。

而且,你给的结构(例如结构体)如果字段已经被赋值了,是无法通过Unmarshal (解组) 去覆盖原值的。

例如:

type Test struct {
  Name string `json:"Name"`
  Age  int    `json:",string,age"`
  S    int    `json:"S"`
}
func main() {
    // 使用的Json编码val是上面代码里的
    // {
    //   "Age": "12",
    //   "Sex": "男",
    //   "成绩": 98
    // }
  var a Test
  _ := json.Unmarshal(val,&a)
  fmt.Printf("%#v\n",a) // main.Test{Name:"", Age:12, S:0}
  var b Test = Test{"12",12,2} // 如果解析到b里 
  _ := json.Unmarshal(val,&b)
  fmt.Printf("%#v\n",a) // main.Test{Name:"12", Age:12, S:2}可以看出来 已经赋值过的字段是不会被覆盖的。
}

要将JSON Unmarshal (解组) 为 interface value,Unmarshal (解组) 将其中一个存储在接口值中:

bool——用于JSON布尔值

float64——用于JSON数字

字符串——用于JSON字符串

[ ]interface{}——用于JSON数组

map[string]interface{}——用于JSON对象 (常用)

---->这就是为啥上面解析到空接口中,出来的却是map[string]interface

nil——表示JSON null

解组到切片Slice

要将JSON数组 Unmarshal (解组) 到切片中,解组将重置切片长度设置为零,然后将每个元素附加到切片。

作为一种特殊情况,要将空JSON数组解组到切片中,解组将使用新的空切片替换切片。

解组到数组Array

要将JSON数组 解组 为Go数组,解组解码JSON数组元素转换为相应的Go数组元素。

如果Go数组小于JSON数组,其他JSON数组元素将被丢弃。

如果JSON数组小于Go数组,附加的Go数组元素设置为零值。

解组到map

要将JSON对象解组到map中,Unmarshal (解组) 首先建立到的map使用。

如果map为nil,则Unmarshal (解组) 分配一个新map。

否则,就Unmarshal (解组) 重用现有map,保留现有条目(即保存你现在已经赋过的值)。

然后Unmarshal (解组) 存储从JSON对象到map的键值对。

|

map的键类型必须为可以是任何字符串类型,也可以是int,implement json.Unmarshaler。

源码中写的:
The map's key type must either be any string type, an integer, 
implement json.Unmarshaler,
or implement encoding.TextUnmarshaler.

如果JSON值不适合给定的目标类型,或者,如果JSON编号溢出了目标类型,则取消编组(Marshal)跳过该字段并尽可能完成Unmarshal (解组)。

如果没有遇到更严重的错误,则Unmarshal (解组)返回描述最早此类错误的Unmarshal (解组)类型错误。

空值解组

JSON空值Unmarshal (解组)为interface{}、map、指针或切片通过将Go值设置为nil。

因为null在JSON中经常用来表示`not present ',将JSON null解组到任何其他Go类型中都无效在该值上,不会产生任何错误。

相关文章
|
14天前
|
运维 监控 算法
监控局域网其他电脑:Go 语言迪杰斯特拉算法的高效应用
在信息化时代,监控局域网成为网络管理与安全防护的关键需求。本文探讨了迪杰斯特拉(Dijkstra)算法在监控局域网中的应用,通过计算最短路径优化数据传输和故障检测。文中提供了使用Go语言实现的代码例程,展示了如何高效地进行网络监控,确保局域网的稳定运行和数据安全。迪杰斯特拉算法能减少传输延迟和带宽消耗,及时发现并处理网络故障,适用于复杂网络环境下的管理和维护。
|
15天前
|
编译器 Go
揭秘 Go 语言中空结构体的强大用法
Go 语言中的空结构体 `struct{}` 不包含任何字段,不占用内存空间。它在实际编程中有多种典型用法:1) 结合 map 实现集合(set)类型;2) 与 channel 搭配用于信号通知;3) 申请超大容量的 Slice 和 Array 以节省内存;4) 作为接口实现时明确表示不关注值。此外,需要注意的是,空结构体作为字段时可能会因内存对齐原因占用额外空间。建议将空结构体放在外层结构体的第一个字段以优化内存使用。
|
15天前
|
存储 缓存 安全
Go 语言中的 Sync.Map 详解:并发安全的 Map 实现
`sync.Map` 是 Go 语言中用于并发安全操作的 Map 实现,适用于读多写少的场景。它通过两个底层 Map(`read` 和 `dirty`)实现读写分离,提供高效的读性能。主要方法包括 `Store`、`Load`、`Delete` 等。在大量写入时性能可能下降,需谨慎选择使用场景。
|
15天前
|
存储 缓存 监控
企业监控软件中 Go 语言哈希表算法的应用研究与分析
在数字化时代,企业监控软件对企业的稳定运营至关重要。哈希表(散列表)作为高效的数据结构,广泛应用于企业监控中,如设备状态管理、数据分类和缓存机制。Go 语言中的 map 实现了哈希表,能快速处理海量监控数据,确保实时准确反映设备状态,提升系统性能,助力企业实现智能化管理。
29 3
|
16天前
|
SQL 安全 Java
阿里双十一背后的Go语言实践:百万QPS网关的设计与实现
解析阿里核心网关如何利用Go协程池、RingBuffer、零拷贝技术支撑亿级流量。 重点分享: ① 如何用gRPC拦截器实现熔断限流; ② Sync.Map在高并发读写中的取舍。
|
17天前
|
存储 算法 安全
基于 Go 语言的公司内网管理软件哈希表算法深度解析与研究
在数字化办公中,公司内网管理软件通过哈希表算法保障信息安全与高效管理。哈希表基于键值对存储和查找,如用户登录验证、设备信息管理和文件权限控制等场景,Go语言实现的哈希表能快速验证用户信息,提升管理效率,确保网络稳定运行。
27 0
|
19天前
|
开发框架 前端开发 Go
eino — 基于go语言的大模型应用开发框架(二)
本文介绍了如何使用Eino框架实现一个基本的LLM(大语言模型)应用。Eino中的`ChatModel`接口提供了与不同大模型服务(如OpenAI、Ollama等)交互的统一方式,支持生成完整响应、流式响应和绑定工具等功能。`Generate`方法用于生成完整的模型响应,`Stream`方法以流式方式返回结果,`BindTools`方法为模型绑定工具。此外,还介绍了通过`Option`模式配置模型参数及模板功能,支持基于前端和用户自定义的角色及Prompt。目前主要聚焦于`ChatModel`的`Generate`方法,后续将继续深入学习。
163 7
|
19天前
|
存储 开发框架 Devops
eino — 基于go语言的大模型应用开发框架(一)
Eino 是一个受开源社区优秀LLM应用开发框架(如LangChain和LlamaIndex)启发的Go语言框架,强调简洁性、可扩展性和可靠性。它提供了易于复用的组件、强大的编排框架、简洁明了的API、最佳实践集合及实用的DevOps工具,支持快速构建和部署LLM应用。Eino不仅兼容多种模型库(如OpenAI、Ollama、Ark),还提供详细的官方文档和活跃的社区支持,便于开发者上手使用。
124 8
|
20天前
|
存储 Go
Go 语言入门指南:切片
Golang中的切片(Slice)是基于数组的动态序列,支持变长操作。它由指针、长度和容量三部分组成,底层引用一个连续的数组片段。切片提供灵活的增减元素功能,语法形式为`[]T`,其中T为元素类型。相比固定长度的数组,切片更常用,允许动态调整大小,并且多个切片可以共享同一底层数组。通过内置的`make`函数可创建指定长度和容量的切片。需要注意的是,切片不能直接比较,只能与`nil`比较,且空切片的长度为0。
Go 语言入门指南:切片
|
20天前
|
存储 算法 Go
Go语言实战:错误处理和panic_recover之自定义错误类型
本文深入探讨了Go语言中的错误处理和panic/recover机制,涵盖错误处理的基本概念、自定义错误类型的定义、panic和recover的工作原理及应用场景。通过具体代码示例介绍了如何定义自定义错误类型、检查和处理错误值,并使用panic和recover处理运行时错误。文章还讨论了错误处理在实际开发中的应用,如网络编程、文件操作和并发编程,并推荐了一些学习资源。最后展望了未来Go语言在错误处理方面的优化方向。