简介
Json(Javascript Object Nanotation)是一种数据交换格式,常用于前后端数据传输。任意一端将数据转换成json 字符串,另一端再将该字符串解析成相应的数据结构,如string类型,strcut对象等。
1.Json Marshal:将数据编码成json字符串
type Stu struct {
Name string `json:"name"`
Age int //来源公众号:「码农编程进阶笔记」
sex string //小写的字段不可导出,所以无法序列化
Class *Class `json:"class"`
}
type Class struct {
Name string
Grade int
}
func main() {
//实例化一个数据结构,用于生成json字符串
stu := Stu{
Name: "张三",
Age: 18,
sex: "男",
}
//指针变量
cla := new(Class)
cla.Name = "1班"
cla.Grade = 3
stu.Class=cla
jsonStu, err := json.Marshal(stu)
if err != nil {
fmt.Println(err.Error())
}
// jsonStu是[]byte类型,转化为string类型便于查看
fmt.Println(string(jsonStu))
}
结果
{"name":"张三","Age":18,"class":{"Name":"1班","Grade":3}}
从结果中可以看出
• 只要是可导出成员(变量首字母大写),都可以转成json。踩坑点:因成员变量sex是不可导出的,故无法转成json。
• 如果变量打上了json标签,如Name旁边的 json:"name" ,那么转化成的json key就用该标签“name”,否则取变量名作为key,如“Age”,“HIgh”。
• bool类型也是可以直接转换为json的value值。Channel, complex 以及函数不能被编码json字符串。当然,循环的数据结构也不行,它会导致marshal陷入死循环。
最后,强调一句:json编码成字符串后就是纯粹的字符串了。
总结: 将map或者结构体传入json.Marshal函数,生成json,返回类型会[]byte
2.Json Unmarshal:将json字符串解码到相应的数据结构
type StuRead struct {
Name interface{} `json:"name"`
Age interface{}
HIgh interface{}
sex interface{}
Class interface{} `json:"class"`
Test interface{}
}
type Class struct {
Name string
Grade int
}
func main() { //json字符中的"引号,需用\进行转义,否则编译出错
//json字符串沿用上面的结果,但对key进行了大小的修改,并添加了sex数据
data:="{\"name\":\"张三\",\"Age\":18,\"high\":true,\"sex\":\"男\",\"CLASS\":{\"naME\":\"1班\",\"GradE\":3}}"
str:=[]byte(data)
//1.Unmarshal的第一个参数是json字符串,第二个参数是接受json解析的数据结构。
//第二个参数必须是指针,否则无法接收解析的数据,如stu仍为空对象StuRead{}
//2.可以直接stu:=new(StuRead),此时的stu自身就是指针 //来源公众号:「码农编程进阶笔记」 stu:=StuRead{}
err:=json.Unmarshal(str,&stu)
//解析失败会报错,如json字符串格式不对,缺"号,缺}等。
if err!=nil{
fmt.Println(err)
}
fmt.Println(stu)
}
结果:
{张三 18 true <nil> map[naME:1班 GradE:3] <nil>}
总结:
*当接收体中存在json串中匹配不了的项时,解析会自动忽略该项,该项仍保留原值。如变量Test,保留空值nil。
若结构体里有的字段不想序列化,而想保留原json,办法还是有的,我们可以将该变量定义为 json.RawMessage类型,保留原json可以进行二次解析。
type StuRead struct {
Name interface{}
Age interface{}
HIgh interface{}
Class json.RawMessage `json:"class"` //注意这里
}