Go语言json包的使用技巧 |Go主题月

简介: 本文整理了一部分我们平时在项目中经常遇到的关于go语言JSON数据与结构体之间相互转换的问题及解决办法。

基本的序列化


json.Marshal(序列化)与json.Unmarshal(反序列化)的基本用法。


type Person struct {
  Name   string
  Age    int64
  Weight float64
}
func main() {
  p1 := Person{
    Name:   "小明",
    Age:    18,
    Weight: 71.5,
  }
  // struct -> json string
  b, err := json.Marshal(p1)
  if err != nil {
    fmt.Printf("json.Marshal failed, err:%v\n", err)
    return
  }
  fmt.Printf("str:%s\n", b)
  // json string -> struct
  var p2 Person
  err = json.Unmarshal(b, &p2)
  if err != nil {
    fmt.Printf("json.Unmarshal failed, err:%v\n", err)
    return
  }
  fmt.Printf("p2:%#v\n", p2)
}
复制代码


输出:


str:{"Name":"小明","Age":18,"Weight":71.5}
p2:main.Person{Name:"小明", Age:18, Weight:71.5}
复制代码


结构体tag介绍


Tag是结构体的元信息,可以在运行的时候通过反射的机制读取出来。


Tag在结构体字段的后方定义,由一对反引号包裹起来,具体的格式如下:


`key1:"value1" key2:"value2"`
复制代码


总结:


  1. 结构体tag由一个或多个键值对组成。键与值使用冒号分隔,值用双引号括起来。
  2. 同一个结构体字段可以设置多个键值对tag,不同的键值对之间使用空格分隔。


使用json tag指定字段名


序列化与反序列化默认情况下使用结构体的字段名,我们可以通过给结构体字段添加tag来指定json序列化生成的字段名。


// 使用json tag指定序列化与反序列化时的行为
type Person struct {
  Name   string `json:"name"` // 指定json序列化/反序列化时使用小写name
  Age    int64
  Weight float64
}
复制代码


忽略某个字段


如果你想在json序列化/反序列化的时候忽略掉结构体中的某个字段,可以按如下方式在tag中添加-。


// 使用json tag指定json序列化与反序列化时的行为
type Person struct {
  Name   string `json:"name"` // 指定json序列化/反序列化时使用小写name
  Age    int64
  Weight float64 `json:"-"` // 指定json序列化/反序列化时忽略此字段
}
复制代码


忽略空值字段


当 struct 中的字段没有值时, json.Marshal() 序列化的时候不会忽略这些字段,而是默认输出字段的类型零值(例如int和float类型零值是 0,string类型零值是"",对象类型零值是 nil)。


如果想要在序列化时忽略这些没有值的字段时,可以在对应字段添加omitempty tag。


举个例子:


type User struct {
  Name  string   `json:"name"`
  Email string   `json:"email"`
  Hobby []string `json:"hobby"`
}
func omitemptyDemo() {
  u1 := User{
    Name: "小明",
  }
  // struct -> json string
  b, err := json.Marshal(u1)
  if err != nil {
    fmt.Printf("json.Marshal failed, err:%v\n", err)
    return
  }
  fmt.Printf("str:%s\n", b)
}
复制代码


输出结果:


str:{"name":"小明","email":"","hobby":null}
复制代码


如果想要在最终的序列化结果中去掉空值字段,可以像下面这样定义结构体:使用omitempty


// 在tag中添加omitempty忽略空值
// 注意这里 hobby,omitempty 合起来是json tag值,中间用英文逗号分隔
type User struct {
  Name  string   `json:"name"`
  Email string   `json:"email,omitempty"`
  Hobby []string `json:"hobby,omitempty"`
}
复制代码


此时,再执行上述的omitemptyDemo,输出结果如下:


str:{"name":"小明"} // 序列化结果中没有email和hobby字段
复制代码


说句题外话,我们使用gorm操作数据库的话,经常会遇到想忽略指定字段修改的问题,比如结构体中的关联实体,只想json展示,form提交时忽略实体,这个问题请关注我的Go语言学习专栏吧。


忽略嵌套结构体空值字段


首先来看几种结构体嵌套的示例:


type User struct {
  Name  string   `json:"name"`
  Email string   `json:"email,omitempty"`
  Hobby []string `json:"hobby,omitempty"`
  Profile
}
type Profile struct {
  Website string `json:"site"`
  Slogan  string `json:"slogan"`
}
func nestedStructDemo() {
  u1 := User{
    Name:  "小明",
    Hobby: []string{"足球", "篮球"},
  }
  b, err := json.Marshal(u1)
  if err != nil {
    fmt.Printf("json.Marshal failed, err:%v\n", err)
    return
  }
  fmt.Printf("str:%s\n", b)
}
复制代码


匿名嵌套Profile时序列化后的json串为单层的:


str:{"name":"小明","hobby":["足球","蓝球"],"site":"","slogan":""}
复制代码


想要变成嵌套的json串,需要改为具名嵌套或定义字段tag:


type User struct {
  Name    string   `json:"name"`
  Email   string   `json:"email,omitempty"`
  Hobby   []string `json:"hobby,omitempty"`
  Profile `json:"profile"`
}
// str:{"name":"小明","hobby":["足球","篮球"],"profile":{"site":"","slogan":""}}
复制代码


想要在嵌套的结构体为空值时,忽略该字段,仅添加omitempty是不够的


type User struct {
  Name     string   `json:"name"`
  Email    string   `json:"email,omitempty"`
  Hobby    []string `json:"hobby,omitempty"`
  Profile `json:"profile,omitempty"`
}
// str:{"name":"小明","hobby":["足球","篮球"],"profile":{"site":"","slogan":""}}
复制代码


还需要使用嵌套的结构体指针


type User struct {
  Name     string   `json:"name"`
  Email    string   `json:"email,omitempty"`
  Hobby    []string `json:"hobby,omitempty"`
  *Profile `json:"profile,omitempty"`  //这里是重点
}
// str:{"name":"小明","hobby":["足球","篮球"]}
复制代码


不修改原结构体忽略空值字段


我们需要json序列化User,但是不想把密码也序列化,又不想修改User结构体,这个时候我们就可以使用创建另外一个结构体PublicUser匿名嵌套原User,同时指定Password字段为匿名结构体指针类型,并添加omitemptytag,示例代码如下:


type User struct {
  Name     string `json:"name"`
  Password string `json:"password"`
}
type PublicUser struct {
  *User             // 匿名嵌套
  Password *struct{} `json:"password,omitempty"`
}
func omitPasswordDemo() {
  u1 := User{
    Name:     "小明",
    Password: "123456",
  }
  b, err := json.Marshal(PublicUser{User: &u1})
  if err != nil {
    fmt.Printf("json.Marshal u1 failed, err:%v\n", err)
    return
  }
  fmt.Printf("str:%s\n", b)  // str:{"name":"小明"}
}
复制代码


优雅处理字符串格式的数字


有时候,前端在传递来的json数据中可能会使用字符串类型的数字,这个时候可以在结构体tag中添加string来告诉json包从字符串中解析相应字段的数据:


type Card struct {
  ID    int64   `json:"id,string"`    // 添加string tag
  Score float64 `json:"score,string"` // 添加string tag
}
func intAndStringDemo() {
  jsonStr1 := `{"id": "1234567","score": "88.50"}`
  var c1 Card
  if err := json.Unmarshal([]byte(jsonStr1), &c1); err != nil {
    fmt.Printf("json.Unmarsha jsonStr1 failed, err:%v\n", err)
    return
  }
  fmt.Printf("c1:%#v\n", c1) // c1:main.Card{ID:1234567, Score:88.5}
}
复制代码


总结


今天只是整理了一部分json的使用技巧,在实际项目中json是不可缺少的一个组成部分,今天立个flag,下一篇会整理gorm相关的使用技巧。

相关文章
|
3月前
|
算法 Java Go
【GoGin】(1)上手Go Gin 基于Go语言开发的Web框架,本文介绍了各种路由的配置信息;包含各场景下请求参数的基本传入接收
gin 框架中采用的路优酷是基于httprouter做的是一个高性能的 HTTP 请求路由器,适用于 Go 语言。它的设计目标是提供高效的路由匹配和低内存占用,特别适合需要高性能和简单路由的应用场景。
324 4
|
4月前
|
Linux Go iOS开发
Go语言100个实战案例-进阶与部署篇:使用Go打包生成可执行文件
本文详解Go语言打包与跨平台编译技巧,涵盖`go build`命令、多平台构建、二进制优化及资源嵌入(embed),助你将项目编译为无依赖的独立可执行文件,轻松实现高效分发与部署。
|
7月前
|
Go 开发者
Go语言包的组织与导入 -《Go语言实战指南》
本章详细介绍了Go语言中的包(Package)概念及其使用方法。包是实现代码模块化、复用性和可维护性的核心单位,内容涵盖包的基本定义、命名规则、组织结构以及导入方式。通过示例说明了如何创建和调用包,并深入讲解了`go.mod`文件对包路径的管理。此外,还提供了多种导入技巧,如别名导入、匿名导入等,帮助开发者优化代码结构与可读性。最后以表格形式总结了关键点,便于快速回顾和应用。
323 61
|
8月前
|
存储 Go
Go语言之接口与多态 -《Go语言实战指南》
Go 语言中的接口是实现多态的核心机制,通过一组方法签名定义行为。任何类型只要实现接口的所有方法即视为实现该接口,无需显式声明。本文从接口定义、使用、底层机制、组合、动态行为到工厂模式全面解析其特性与应用,帮助理解 Go 的面向接口编程思想及注意事项(如 `nil` 陷阱)。
239 22
|
6月前
|
JSON 中间件 Go
Go语言实战指南 —— Go中的反射机制:reflect 包使用
Go语言中的反射机制通过`reflect`包实现,允许程序在运行时动态检查变量类型、获取或设置值、调用方法等。它适用于初中级开发者深入理解Go的动态能力,帮助构建通用工具、中间件和ORM系统等。
344 63
|
7月前
|
测试技术 程序员 Go
Go语言测试简明指南:深度解读go test命令
总的来说,go test是 Go 语言中一个强而有力的工具,每个 Go 程序员都应该掌握并把它融入到日常的开发和调试过程中。就像是一个眼镜过滤出的太阳,让我们在宽阔的代码海洋中游泳,而不是淹没。用好它,让我们的代码更健壮,让我们的生产力更强效。
587 23
|
7月前
|
数据可视化 测试技术 Go
Go 语言测试与调试:`go test` 工具用法
`go test` 是 Go 语言内置的测试工具,支持单元测试、基准测试、示例测试等功能。本文详解其常用参数、调试技巧及性能测试命令,并提供实际项目中的应用示例与最佳实践。
|
7月前
|
测试技术 Go 开发者
Go语言常见接口设计技巧-《Go语言实战指南》
本文分享了 Go 语言中接口设计的最佳实践与技巧。首先介绍了接口设计原则,包括面向接口编程和接口隔离原则(定义最小化接口)。接着详细讲解了常用技巧:关注行为而非数据、优先返回接口隐藏实现细节、遵循“-er”命名惯例、使用接口组合提升灵活性、通过 Mock 接口简化单元测试,以及避免导出仅内部使用的接口。最后以表格形式总结了各技巧的核心要点,帮助开发者编写更清晰、可维护的代码。
273 11
|
7月前
|
缓存 安全 Go
Go语言依赖管理与版本控制-《Go语言实战指南》
本章深入探讨Go语言中的依赖管理与版本控制,重点介绍Go Modules的使用方法。内容涵盖依赖管理的重要性、语义化版本控制(SemVer)、查看和管理依赖版本、主版本路径规则、常见操作场景、国内代理加速、依赖安全(go.sum文件)、版本冲突解决及版本锁定与回退等主题。通过学习,读者将掌握如何实现清晰、稳定且可重复构建的项目依赖管理。
|
8月前
|
JSON JavaScript 前端开发
Go语言JSON 序列化与反序列化 -《Go语言实战指南》
本文介绍了 Go 语言中使用 `encoding/json` 包实现 JSON 与数据结构之间的转换。内容涵盖序列化(`Marshal`)和反序列化(`Unmarshal`),包括基本示例、结构体字段标签的使用、控制字段行为的标签(如 `omitempty` 和 `-`)、处理 `map` 和切片、嵌套结构体序列化、反序列化未知结构(使用 `map[string]interface{}`)以及 JSON 数组的解析。最后通过表格总结了序列化与反序列化的方法及类型要求,帮助开发者快速掌握 JSON 数据处理技巧。