Go实现json字符串与各类struct相互转换

简介: 文章通过Go语言示例代码详细演示了如何实现JSON字符串与各类struct之间的相互转换,包括结构体对象生成JSON字符串和JSON字符串映射到struct对象的过程。

不废话了都在代码中了


package main

import (
   "fmt"
   "reflect"
   "encoding/json"
   "strings"
)

type Class struct {
   Grade int `json:"grade"` //年级
   ClassNumber int  `json:"classNumber"` //班级号
}

type Student struct{
   Name string  //大写开头,可被导出,没有`json:".."`,导出json的字段名是原本名称
   age int  //小写开题,不可被导出
   Hight int `json:"currentHight"` //导出对应json的字段名为currentHight
   Class *Class `class` //指针,指向引用对象;如果不用指针,只是值复制
}

func doMarshal(){//对象转json字符串
   nClass:=new(Class)//new只给给特定类型分配内存,设置“零”值,返回其地址(指针)
   fmt.Printf("nClass的类型是%s,内容是%v\n",reflect.TypeOf(nClass),*nClass)
   nClass.Grade=3
   nClass.ClassNumber=6
   nStudents:=make([]*Student,0)
   //make只用于map,slice和channel,并且不显示返回指针
   //这个切片,存放Student的指针
   nStudent:=Student{"Lily",7,116,nClass}
   jsonBytes,err1:=json.Marshal(nStudent)//解析后的是[]byte
   if err1!=nil{
      fmt.Printf("转json失败:%v\n",err1)
      return
   }
   fmt.Println("转成的JSON:") //age不会被导出
   //{"Name":"Lily","currentHight":116,"Class":{"grade":3,"classNumber":6}}
   fmt.Println(string(jsonBytes))
   nStudents=append(nStudents,&Student{"Lilei",8,130,nClass})
   nStudents=append(nStudents,&nStudent)
   josnListBytes,err2:=json.Marshal(nStudents)
   if err2!=nil{
      fmt.Printf("转jsonList失败:%v\n",err2)
      return
   }
   fmt.Println("转成的列表型JSON:")
   fmt.Println(string(josnListBytes))
   //[{"Name":"Lilei","currentHight":130,"Class":{"grade":3,"classNumber":6}},{"Name":"Lily","currentHight":116,"Class":{"grade":3,"classNumber":6}}]
}

func doUnMarshal(){//json字符串转对象
   jsonStr:=`
      {
         "Name":"Lily",
         "currentHight":116,
         "age":12,
         "Class":{
            "grade":3,
            "classNumber":6
         },
         "score":[98,100,95]
      }
   `
   jsonListStr:=`[
      {
         "Name":"Lucy",
         "currentHight":120,
         "Class":{
            "grade":3,
            "classNumber":5
         }
      },
      {
         "Name":"Lily",
         "currentHight":116,
         "Class":{
            "grade":3,
            "classNumber":6
         }
      }
   ]`
   //第一种解析json方式,解析到Struct/[]Struct
   student:=Student{}//同new(Student)
   err:=json.Unmarshal([]byte(jsonStr),&student)
   //Unmarshall第2个参数必须是指针,否则报错:json: Unmarshal(non-pointer main.Student)
   //因为必须解析到具体的对象,所以需传入对象引用,而不是值传递
   //score在Student中没有此字段,所以被忽略了
   if err!=nil{
      fmt.Printf("解析json字符串异常:%s\n",err)
   }
   fmt.Printf("学生的名称是%v,班级信息是%v,年龄是%v(私有对象不能导入,初始为0)\n",student.Name,*student.Class,student.age)
   //学生的名称是Lily,学生的班级信息是{3 6},学生的年龄是0
   students:=[]*Student{} //定义切片,同make([]*Student,0)
   err=json.Unmarshal([]byte(jsonListStr),&students)
   if err!=nil{
      fmt.Printf("解析json字符串异常:%s\n",err)
   }
   for _,stu:=range students{ //这里stu是指针类型,获取其属性可以直接用.Name,也可以解引用后用.Name
      fmt.Printf("列表:学生的名称是%s,身高是%d,在%d年级%d班\n",stu.Name,(*stu).Hight,(*stu.Class).Grade,stu.Class.ClassNumber)
   }
   //第二种解析到interface{}/[]interface{}
   fmt.Println("*************解析json*************")
   var student1 interface{}
   err=json.Unmarshal([]byte(jsonStr),&student1)
   if err!=nil{
      fmt.Printf("解析json字符串异常:%s\n",err)
   }
   c:=-1
   resolve2JosnObj(student1,c)
   /*
   *************解析json*************
   map元素:
   map[Name]的元素: 类型是string,值是 Lily
   map[currentHight]的元素: 类型float64,值是 116
   map[age]的元素: 类型float64,值是 12
   map[Class]的元素: map元素:
   ---map[classNumber]的元素: 类型float64,值是 6
   ---map[grade]的元素: 类型float64,值是 3
   map[score]的元素: list元素:
   ---第0个元素: 类型float64,值是 98
   ---第1个元素: 类型float64,值是 100
   ---第2个元素: 类型float64,值是 95
    */
   fmt.Println("*************解析jsonlist*************")
   var students1 interface{}
   err=json.Unmarshal([]byte(jsonListStr),&students1)
   if err!=nil{
      fmt.Printf("解析jsonlist字符串异常:%s\n",err)
   }
   d:=-1
   resolve2JosnObj(students1,d)
   /*
   *************解析jsonlist*************
   list元素:
   第0个元素: map元素:
   ---map[Name]的元素: 类型是string,值是 Lucy
   ---map[currentHight]的元素: 类型float64,值是 120
   ---map[Class]的元素: map元素:
   ------map[grade]的元素: 类型float64,值是 3
   ------map[classNumber]的元素: 类型float64,值是 5
   第1个元素: map元素:
   ---map[Class]的元素: map元素:
   ------map[grade]的元素: 类型float64,值是 3
   ------map[classNumber]的元素: 类型float64,值是 6
   ---map[Name]的元素: 类型是string,值是 Lily
   ---map[currentHight]的元素: 类型float64,值是 116
    */
}

func resolve2JosnObj(objI interface{},c int){
   c=c+1
   switch obj:=objI.(type) { //此处[interface{}].(type) 专门用于switch的类型判断
   case string:
      fmt.Println("类型是string,值是",obj)
   case float64:
      fmt.Println("类型float64,值是",obj)
   case map[string]interface{}:
      fmt.Println("map元素:")
      for k,vi:=range obj{
         fmt.Printf("%smap[%s]的元素: ",strings.Repeat("---",c),k)
         resolve2JosnObj(vi,c)
      }
   case []interface{}:
      fmt.Println("list元素:")
      for i,vi:=range obj{
         fmt.Printf("%s第%d个元素: ",strings.Repeat("---",c),i)
         resolve2JosnObj(vi,c)
      }

   default:
      fmt.Println("无法判断类型,类型是",reflect.TypeOf(obj),"值是",obj)

   }

}

func main() {
   doMarshal()//对象转json字符串
   doUnMarshal()
}

简单总结

1、结构体对象可生成json字符串,Marshal()是[]byte,需要string去转换
2、json字符串可以映射到一个struct,但仅限公共元素(大写开头);也可通用的转换到空接口interfece[],使用对应转换到需要的内容

相关文章
|
4月前
|
人工智能 自然语言处理 算法
Go语言统计字符串中每个字符出现的次数 — 简易频率分析器
本案例实现一个字符统计程序,支持中文、英文及数字,可统计用户输入文本中各字符的出现次数,并以整洁格式输出。内容涵盖应用场景、知识点讲解、代码实现与拓展练习,适合学习文本分析及Go语言基础编程。
|
4月前
|
JSON 人工智能 Go
在Golang中序列化JSON字符串的教程
在Golang中,使用`json.Marshal()`可将数据结构序列化为JSON格式。若直接对JSON字符串进行序列化,会因转义字符导致错误。解决方案包括使用`[]byte`或`json.RawMessage()`来避免双引号被转义,从而正确实现JSON的序列化与反序列化。
183 7
|
6月前
|
JSON Go C语言
Go语言之定义结构体(Struct)-《Go语言实战指南》
Go 语言中的结构体(`struct`)是一种复合数据类型,可将多个不同类型的字段组合成一个类型。本文介绍了结构体的基本定义、实例创建方式、字段访问与修改、零值特性、比较规则、嵌套使用及标签功能。通过示例代码详细讲解了如何定义和操作结构体,以及其在 JSON 编码等场景的应用。
|
12月前
|
存储 Go 索引
go语言中遍历字符串
go语言中遍历字符串
196 5
|
6月前
|
JSON JavaScript 前端开发
Go语言JSON 序列化与反序列化 -《Go语言实战指南》
本文介绍了 Go 语言中使用 `encoding/json` 包实现 JSON 与数据结构之间的转换。内容涵盖序列化(`Marshal`)和反序列化(`Unmarshal`),包括基本示例、结构体字段标签的使用、控制字段行为的标签(如 `omitempty` 和 `-`)、处理 `map` 和切片、嵌套结构体序列化、反序列化未知结构(使用 `map[string]interface{}`)以及 JSON 数组的解析。最后通过表格总结了序列化与反序列化的方法及类型要求,帮助开发者快速掌握 JSON 数据处理技巧。
|
7月前
|
Go 索引
【LeetCode 热题100】394:字符串解码(详细解析)(Go语言版)
本文详细解析了 LeetCode 热题 394:字符串解码。题目要求对编码字符串如 `k[encoded_string]` 进行解码,其中 `encoded_string` 需重复 `k` 次。文章提供了两种解法:使用栈模拟和递归 DFS,并附有 Go 语言实现代码。栈解法通过数字栈与字符串栈记录状态,适合迭代;递归解法则利用函数调用处理嵌套结构,代码更简洁。两者时间复杂度均为 O(n),但递归需注意栈深度问题。文章还总结了解题注意事项及适用场景,帮助读者更好地掌握字符串嵌套解析技巧。
176 6
|
8月前
|
存储 机器学习/深度学习 缓存
🚀 力扣热题 394:字符串解码(详细解析)(Go语言版)
文章提供了两种解法:栈结构和递归解法。栈解法通过维护数字栈与字符串栈,依次处理 `[` 和 `]`,构造解码结果;递归解法则利用函数调用逐层解析嵌套结构。两者时间复杂度均为 $O(n)$,空间复杂度也为 $O(n)$。栈解法直观易懂,适合初学者;递归解法优雅简洁,适合处理深度嵌套规则。掌握这两种方法,可灵活应对类似问题,提升解题能力。
245 11
|
11月前
|
程序员 Go
go语言中结构体(Struct)
go语言中结构体(Struct)
324 71
|
11月前
|
JSON 数据格式 C++
cpp struct json相互转换
使用 `nlohmann/json`库,可以简便地在C++中实现结构体与JSON之间的相互转换。通过定义序列化和反序列化函数,可以处理简单和复杂的结构体类型。这种方法不仅提高了代码的可读性和维护性,还增强了数据交换的灵活性。
561 26
|
9月前
|
JSON JavaScript 前端开发
处理从API返回的JSON数据时返回Unicode编码字符串怎么处理
在处理API返回的JSON数据时,遇到类似`\u7f51\u7edc\u8fde\u63a5\u9519\u8bef`的Unicode编码字符串,可使用JavaScript内置方法转换为可读文字。主要方法包括:1. 使用`JSON.parse`自动解析;2. 使用`decodeURIComponent`和`escape`组合解码;3. 在API调用中直接处理响应数据。这些方法能有效处理多语言内容,确保正确显示非ASCII字符。

热门文章

最新文章