JSON(Javascript Object Notation)是一种轻量级的数据交换语言,以文字为基础,具有自我描述性且易于让人阅读。尽管JSON是Javascript的一个子集,但JSON是独立于语言的文本格式,并且采用了类似于C语言家族的一些习惯。JSON与XML最大的不同在于XML是一个完整的标记语言,而JSON不是。JSON由于比XML更小、更快,更易解析,以及浏览器的内建快速解析支持,使得其更适用于网络数据传输领域。

1. 标准库解析

Go语言标准库已经支持JSON,encoding/json。

func Marshal(v interface{}) ([]byte, error)

func Unmarshal(data []byte, v interface{}) error

分别实现结构体或interface转JSON字符串,JSON字符串转结构体或interface的操作。

Marshal函数只有在转换成功的时候才会返回数据,在转换的过程中我们需要注意几点:

·         JSON对象只支持string作为key,所以要编码一个map,那么必须是map[string]T这种类型(T是Go语言中任意的类型)

·         Channel, complex和function是不能被编码成JSON的

·         嵌套的数据是不能编码的,不然会让JSON编码进入死循环

·         指针在编码的时候会输出指针指向的内容,而空指针会输出null

1.1 结构体解析

JSON字符串转化为Struct时,struct的字段名与JSON的key对应,如果JSON的key是Foo,怎样找到对应的字段呢?

·         首先查找tag含有Foo的可导出的struct字段(首字母大写)

·         其次查找字段名是Foo的导出字段

·         最后查找类似FOO或者FoO这样的除了首字母之外其他大小写不敏感的导出字段

能够被赋值的字段必须是可导出字段(即首字母大写)。同时JSON解析的时候只会解析能找得到的字段,找不到的字段会被忽略。这样的一个好处是:当你接收到一个很大的JSON数据结构而你却只想获取其中的部分数据的时候,你只需将你想要的数据对应的字段名大写,即可轻松解决这个问题。

注:字符串转struct时,结构体与JSON子串实现双向最大匹配,只解析能解析的子串。

struct转JSON时,可用tag指定key名称:

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

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

·         tag中如果带有",omitempty"选项,那么如果该字段值为空,就不会输出到JSON串中(,后无空格),只有struct转json时起作用。

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

package main

import (
"encoding/json"
"fmt"
) type Server struct{
ServerName string `json:"serverName"`
ServerIP string `json:"serverIP"`
Name string `json:"-"`
} type Serverslice struct {
Servers [] Server `json:"servers"`
StrFirst string `json:"strFirst"`
StrSecond string `json:"strSecond,string"`
} func main(){
var s Serverslice
s.StrFirst = `Go "1.0" `
s.StrSecond = `Go "1.0" `
s.Servers = append(s.Servers, Server{ServerName: "Shanghai_VPN", ServerIP: "127.0.0.1", Name:"wang"})
s.Servers = append(s.Servers, Server{ServerName: "Beijing_VPN", ServerIP: "127.0.0.2"})
b, err := json.Marshal(s)
if err != nil {
fmt.Println("json err:", err)
}
fmt.Println(string(b)) fmt.Println("---------------Unmarshal-----------")
str := `{"Name":"Wednesday", "Age":, "Parents":["Gomez", "Morticia"]}`
var f interface{}
err = json.Unmarshal([]byte(str), &f)
if err != nil {
return
}
fmt.Println(":", f) if m, ok := f.(map[string]interface{}); ok {
for k, v := range m {
switch vv := v.(type){
case string:
fmt.Println(k, "is string", vv)
case int:
fmt.Println(k, "is int", vv)
case float64:
fmt.Println(k, "is float64", vv)
case []interface{}:
fmt.Println(k, "is an array:")
for i, u := range vv {
fmt.Println(i, u)
}
default:
fmt.Println(k, "is of a type that I don't know how to handle")
}
}
}
}

运行结果:

$ go run assemble.go
{"servers":[{"serverName":"Shanghai_VPN","serverIP":"127.0.0.1"},{"serverName":"Beijing_VPN","serverIP":"127.0.0.2"}],"strFirst":"Go \"1.0\" ","strSecond":"\"Go \\\"1.0\\\" \""}
---------------Unmarshal-----------
: map[Name:Wednesday Age: Parents:[Gomez Morticia]]
Name is string Wednesday
Age is float64
Parents is an array:
Gomez
Morticia

result

1.2 interface解析

上述代码中JSON字符串解析到interface中,通过断言输出各个数据值。

1.3 空值处理

struct转json子串

当结构体tag指定",omitempty"时,空值会被舍弃,不输出到json串口。

  • bool空值为false
  • string的空值为""(字符串长度为0)
  • 切片类型为nil
  • 指针类型为nil

注:struct在golang为值类型,零值为结构体中各成员的零值,所以当结构体未赋值时(struct{}),json串输出{},而非被舍弃,即使设置omitempty。

json子串转struct

字符串转struct时,结构体与JSON子串实现双向最大匹配,只解析能解析的子串。

Json串转换为strunt后,因为struct中各成员都必须有值,那如何判定这个值是字符串输入的还是golang赋的默认初值呢???

1)比如bool,如果值整好是false,如何判断这个false是json子串输入的(因为bool的key不论字符串是否包含false的key,转换为struct后都是false)?

不能判断。参考:go json 字符串转struct时 如何判断 bool类型的字段是否存在?

go 初始struct对象的时候是默认初始化里边的字段的。这意味着你不能根据json 中bool字段的存在来对应影响生成的struct对象。

要不你把json的bool字段换成默认的go语义为空的数据类型,比如string(有局限)。或者你override go中转换struct对象的方法。

或者 可以把struct中的bool类型改为指针类型 *bool,这样如果不存在这个字段,那么就是nil,否则就是存在,取出值即可(有局限)。

2)nil空值,如果是nil,json中肯定没有输入,如果有输入就非nil了。但是如果json串中nil的key本身就是null,依然无法判断(有局限)。

3)string的“”,如果json串中的“”的key本身值就是“”,依然无法判断(有局限)。

4)struct的null,如果本身输入的就是null,依然无法判断(有局限)。

(备注:

如此就要求json串中不设置的值,就不要填写,特别是null,false或“”。

这样可通过与空值比较来判断是否有输入,从而清空原来的保存的数据(http的patch)。

特别地,struct如果是可比较的,空的struct应该与struct{}比较。)

1.4 默认值设置

struct转换为json串时,切片的默认值是null,如何输出[]呢?

参考:Go语言设置JSON的默认值

给需要设置的JSON字段初试化你想设置的值就OK。

比如我想让[]string类型的字段的默认值是[],而不是nil,那我就make([]string, 0)赋值给该字段。转成JSON输出后,就是[]

struct test{
Test2 []string
} t.Test2 = make([]string, )

1.5 不修改结构体情况下忽略特定字段

将字段首字母改为小写或添加 json:"-" 标签能够在 json.Marshal() 时忽略指定字段,但此处讨论的是在不修改原 struct 结构的前提下过忽略部分字段的方法。

参考:golang struct 转 json 时如何忽略部分字段

1)定义新 struct,在新 struct 中只保留必要字段。

2)定义 tag 不同结构相同的 struct。

3)通过map转换。

2. 其他库解析

go-simplejson

bitly公司开源了一个叫做simplejson的包,在处理未知结构体的JSON时相当方便。

package main

import (
"fmt"
sj "github.com/bitly/go-simplejson"
) func main(){
js, err := sj.NewJson([]byte(`{
"test":{
"array":[, "", ],
"int": ,
"float": 1.10,
"bignum": ,
"string": "simplejson",
"bool": true
}
}`))
if err != nil {
return
} arr, _ := js.Get("test").Get("array").Array()
i, _ := js.Get("test").Get("int").Int()
ms := js.Get("test").Get("string").MustString() fmt.Println("arr", arr, "i", i, "ms", ms)
}

运行结果:

arr [  ] i  ms simplejson

easyjson

https://github.com/mailru/easyjson.git

encoding/json采用反射设计,easyjson采用生成代码的方式(针对特定json),非常高效。

参考:

1. Go Web编程 7.2 JSON处理

2. https://github.com/bitly/go-simplejson

3. https://github.com/mailru/easyjson.git

4. go json转换实践中遇到的坑   整数变浮点数问题   时间格式

5. golang struct 转 json 时如何忽略部分字段  https://www.dazhuanlan.com/2019/10/15/5da55aaccfa55/

JSON的Go解析的更多相关文章

  1. C# Json传值与解析

    最近接触了工作室的项目,觉得一个功能的实现有点不好,心想不能就动手改了下,做了才知道我的js是多么的渣,功能是这样的: 我要实现的功能就是当选择学院时,就放松get请请求到后台,后台返回json信息再 ...

  2. android Json Gson FastJson 解析

    一 Json xml <?xml version="1.0" encoding="utf-8"?> <LinearLayout xmlns:a ...

  3. JS中将JSON的字符串解析成JSON数据格式《转》

    在JS中将JSON的字符串解析成JSON数据格式,一般有两种方式: 1.一种为使用eval()函数. 2. 使用Function对象来进行返回解析. 使用eval函数来解析,并且使用jquery的ea ...

  4. android基础---->JSON数据的解析

    上篇博客,我们谈到了XML两种常用的解析技术,详细可以参见我的博客(android基础---->XMl数据的解析).网络传输另外一种数据格式JSON就是我们今天要讲的,它是比XML体积更小的数据 ...

  5. Android 之 json数据的解析(jsonReader)

    json数据的解析相对而言,还是比较容易的,实现的代码也十分简单.这里用的是jsonReade方法来进行json数据解析. 1.在解析之前,大家需要知道什么是json数据. json数据存储的对象是无 ...

  6. 《项目经验》--后台一般处理程序向前台JS文件传递JSON,JS解析JSON,将数据显示在界面--显示在DropDownList 或 显示在动态创建的table中

    http://blog.csdn.net/mazhaojuan/article/details/8599167 先看一下我要实现的功能界面: 这篇文章主要介绍:后台一般处理程序把从数据库查找的数据,转 ...

  7. iOS开发网络篇-JSON文件的解析

    一.什么是JSON数据 1.JSON的简单介绍 JSON:是一种轻量级的传输数据的格式,用于数据的交互 JSON是javascript语言的一个子集.javascript是个脚本语言(不需要编译),用 ...

  8. iOS开发网络篇—JSON数据的解析

    iOS开发网络篇—JSON数据的解析 iOS开发网络篇—JSON介绍 一.什么是JSON JSON是一种轻量级的数据格式,一般用于数据交互 服务器返回给客户端的数据,一般都是JSON格式或者XML格式 ...

  9. Android Json生成及解析实例

    JSON的定义: 一种轻量级的数据交换格式,具有良好的可读和便于快速编写的特性.业内主流技术为其提供了完整的解决方案(有点类似于正则表达式 ,获得了当今大部分语言的支持),从而可以在不同平台间进行数据 ...

  10. Json,Gson,FastJson解析笔记

    Json,Gson,FastJson解析笔记 1.将JavaBean转换成Json对象: public static String CreatJsonFromObject(Object key,Obj ...

随机推荐

  1. 【视频开发】ONVIF客户端搜索设备获取rtsp地址开发笔记(精华篇)

    转载地址:http://blog.csdn.net/gubenpeiyuan/article/details/25618177 概要:           目前ONVIF协议家族设备已占据数字监控行业 ...

  2. 【转】Object.keys方法之详解

     一.语法 Object.keys(obj) 参数:要返回其枚举自身属性的对象 返回值:一个表示给定对象的所有可枚举属性的字符串数组 二.处理对象,返回可枚举的属性数组 let person = {n ...

  3. noi openjudge 1768:最大子矩阵

    链接:http://noi.openjudge.cn/ch0406/1768/ 描述已知矩阵的大小定义为矩阵中所有元素的和.给定一个矩阵,你的任务是找到最大的非空(大小至少是1 * 1)子矩阵. 比如 ...

  4. jquery如何生成图片验证码

    jQuery(function($){ /**生成一个随机数**/ function randomNum(min, max) { return Math.floor(Math.random() * ( ...

  5. Ubuntu搭建交叉编译开发环境

    在Linux驱动开发过程中,往往需要搭建交叉编译开发环境,其中,最重要的环节就是安装交叉编译工具链,本文介绍如何在Ubuntu下搭建交叉编译开发环境. 1.官网下载交叉编译工具链 链接如下: http ...

  6. 【手写代码】计算1-n中总共有多少二进制1

    #include<bits/stdc++.h> #include<vector> using namespace std; //时间复杂度:O(N) int f(int x) ...

  7. DatabaseGeneratedOption

    [DatabaseGenerated(DatabaseGeneratedOption.Identity)] 添加时  获取值   自增  默认值,,后期无法修改如:Id(AUTO_INCREMENT, ...

  8. JFR 使用记录

    进程的内存信息,可以使用jmap 和 jstack 等dump出文件,使用jhat 分析 dump 文件.不过比较简陋. 可以不停进程的方式有 JFR 或者taobao 开源组件. 本篇只记录JFR相 ...

  9. 英语insuraunce保险

    中文名:保险 外文名:insurance或insuraunce 类型:保障机制,商业行为 作用:资金融通.损失补偿等 原则:分摊.代位.大数法则等原则 性质:契约经济关系 意义:市场经济条件下风险管理 ...

  10. Robot Arms AtCoder - 4432 (构造)

    大意: 给定平面上$n$个点$(x_i,y_i)$. 要求构造一个序列$d$, $d_i$表示每步走的距离, 再构造$n$个命令串, 要求从原点出发按照第$i$个命令走, 走完恰好到达$(x_i,y_ ...