Go语言编程系列文章
Go语言学习历程整理:
序号 | 文章 |
1. | 👀 |
文章目录
前言JSON是一种比XML更轻量级的数据交换格式,易于阅读和编写的同时,也易于程序的解析和生成。此外,虽然JSON源于JS, 由于JSON采用了完全独立于语言的文本格式,采用键值的方式构建描述信息,因此逐渐成为较理想的、跨平台的、跨语言的数据交换格式。
Go语言对JSON提供的很好的支持:GO语言内置encoding/json标准库。开发人员可以通过标准库函数轻松完成JSON数据格式的生成和解析工作。
下面对JSON文件的生成和解析做一个简单介绍。
一、🔔将数据编码为JSON格式Go原因中可以使用 json.Marshal() 函数将一组数据进行JSON格式编码。它的原型如下:
func Marshal(v interface{}) ([]byte, error)
接收任意类型的参数,并将其转换为json格式,最后通过byte切片的方式返回。
用法如下:
package main
import (
"encoding/json"
"fmt"
)
type Student struct {
Name string `json:"name"`
Age int `json:"age"`
Email string `json:"email"`
Wechat string `json:"wechat"`
}
func JsonMarshal() {
xiaoMing := Student{
Name: "xiaoMing",
Age: 23,
Email: "vip_13031075266@163.com",
Wechat: "ssjjqq23",
}
fmt.Println(xiaoMing)
data, _ := json.Marshal(xiaoMing)
fmt.Printf("%s\n", data)
}
运行结果如下:
{"name":"xiaoMing","age":23,"email":"vip_13031075266@163.com","wechat":"ssjjqq23"}
在Go中, JSON转化前后的数据类型映射如下。
0️⃣ 布尔值转化为JSON后还是布尔类型。
1️⃣ 浮点数和整型会被转化为JSON里边的常规数字。
2️⃣ 字符串将以UTF-8编码转化输出为Unicode字符集的字符串。
3️⃣ 数组和切片会转化为JSON里边的数组,但[]byte类型的值将会被转化为 Base64 编码后
的字符串, slice类型的零值会被转化为 null。
4️⃣ 结构体会转化为JSON对象,并且只有结构体里边以大写字母开头的可被导出的字段才会
被转化输出,而这些可导出的字段会作为JSON对象的字符串索引。
5️⃣ 转化一个map类型的数据结构时,该数据的类型必须是 map[string]T(T可以是
encoding/json 包支持的任意数据类型)。
在Go中使用 json.Unmarshal()函数将JSON格式数据解码为Go中期望的数据结构。原型如下:
func Unmarshal(data []byte, v interface{}) error
data : json编码的数据信息
v : 解码后的Go中的数据结构
用法如下:
package main
import (
"encoding/json"
"fmt"
"log"
)
type Student struct {
Name string `json:"name"`
Age int `json:"age"`
Email string `json:"email"`
Wechat string `json:"wechat"`
}
type Msg struct {
MessageType string `json:"message_type"`
MessageBody json.RawMessage `json:"message_body"`
}
func JsonUnMarshal() {
jsonData := []byte(`{"name":"xiaoMing","age":23,
"email":"vip_13031075266@163.com","wechat":"ssjjqq23"}`)
var xiaoMing Student
err := json.Unmarshal(jsonData, &xiaoMing)
if err != nil {
log.Print(err)
}
fmt.Printf("%+v\n", xiaoMing)
}
运行结果如下:
{Name:xiaoMing Age:23 Email:vip_13031075266@163.com Wechat:ssjjqq23}
🍓🍓🍓注意:
如果JSON中的字段在Go的目标类型中没有,json.Unmarshal()函数会在解码过程中将其丢弃。因此可以实现从JSON格式中获取直接字段的数据信息。
三、🔔解码未知结构的JSON数据在使用 json.Unmarshal()函数解析JSON格式数据时,需要传入目标类型的对象地址,而对于位置结构的JSON数据,由于无法提前得知目标类型,因此需要特别说明。
不过,值得庆幸的是: Go中提供了接口类型。而接口是通用类型,可以表示任意类型的。因此面对位置结构的json数据时,我们可以通过传入一个接口类型的变量来完成数据的解析。
在解码JSON数据的过程中,JSON数据里面的元素转换原则如下:
✅ JSON中的布尔值将会转换为Go中的bool类型
✅ 数值会被转换为Go中的float64类型
✅ 字符串转换后还是string类型
✅ JSON数组会转换为[]interface{}类型
✅ JSON对象会转换为map[string]interface{}类型
✅ null值会转换为nil
常见使用方法如下:
func JsonUnMarshalUnknownJson() {
info := []byte(`{"MESSAGE_type": "tcp_log",
"message_body": {"time": 1629344853314, "sess_id": "0x611DD45400000001",
"sport": 12378, "dport": 21, "sip": "10.43.80.78", "dip": "10.44.98.224",
"smac": "30:9C:23:36:97:00", "dmac": "80:05:88:CA:54:16", "device_port_id": 1,
"close_status": "FIN", "device_id": "200707014084", "data_type": 1,
"interface_icon": "hahah", "network_protocol": 0, "transport_protocol": 6,
"session_protocol": 0, "app_protocol": 10, "log_type": 1, "client_total_byte": 126,
"client_total_pkt": 17, "server_total_byte": 437, "server_total_pkt": 19,
"flow_start_time": 1629344852098, "flow_end_time": 1629344852315,
"flow_duration": 217, "avg_pkt_byte": 0, "avg_pkt_num": 0,
"total_byte": 563, "avg_pkt_size": 15, "avg_delay_time": 244,
"retrans_pkt_num": 0}}`)
var unknowInfo interface{}
err := json.Unmarshal(info, &unknowInfo)
if err != nil {
fmt.Printf("json unmarshal error:%s\n", err)
return
}
msg, ok := unknowInfo.(map[string]interface{})
if ok {
for k, v := range msg {
switch v2 := v.(type) {
case string:
fmt.Println(k, "is string", v2)
case int:
fmt.Println(k, "is int", v2)
case bool:
fmt.Println(k, "is bool", v2)
case []interface{}:
fmt.Println(k, "is an array:")
for i, iv := range v2 {
fmt.Println(i, iv)
}
default:
fmt.Println(k, "is another type!!!")
}
}
}
}
总结
文章主要说明了Go中对JSON格式的编码和解码的使用。