如何控制Go编码JSON数据时的行为(问题及解决方案)

今天来聊一下我在Go中对数据进行 JSON 编码时遇到次数最多的三个问题以及解决方法,大家来看看是不是也为这些问题挠掉了不少头发。

自定义JSON键名

这个问题加到文章里我是有所犹豫的,因为基本上大家都会,不过属于同类问题我还是放进来了,对新接触 Go 的同学更友好些。

我们先从最常见的一个问题说,首先在Go 程序中要将数据编码成JSON 格式时通常我们会先定义结构体类型,将数据存放到结构体变量中。

type Address struct {
 Type string
 City string
 Country string
}

type CreditCard struct {
 FirstName string
 LastName string
 Addresses []*Address
 Remark string
}

home := &Address{"private", "Aartselaar", "Belgium"}
office := &Address{"work", "Boom", "Belgium"}
card := VCard{"Jan", "Kersschot", []*Address{home, office}, "none"}

js, err := json.Marshal(card)
fmt.Printf("JSON format: %s", js)

只有导出的结构体成员才会被编码,这也就是我们为什么选择用大写字母开头的字段名称。在编码时,默认使用结构体字段的名字作为JSON对象中的 key ,但是一般JSON 是给 HTTP接口 返回数据使用的,在接口的规范里针对数据我们一般都要求返回 snake case 风格的字段名。解决这个问题的方法是在结构体声明时在结构体字段标签里可以自定义对应的 JSON key

所以我们把结构体声明改为如下即可:

type Address struct {
 Type string `json:"type"`
 City string `json:"city"`
 Country string `json:"country"`
}

编码JSON时忽略掉指定字段

并不是所有数据我们都期望编码到 JSON 中暴露给外部接口的,所以针对一些敏感的字段我们往往希望将其从编码后的 JSON 数据中忽略掉。那么上面也说了只有导出的结构体成员才会被编码,有的同学会问我直接用小写的字段名不行吗?可是未导出字段只能在包内访问,像这种携带内部敏感数据的往往都是应用的基础数据,由项目的公共包来提供的。那么怎么既能维持字段的导出性又能让其在 JSON 数据中被忽略掉呢? 还是使用结构体的标签进行注解,比如下面定义的结构体,可以把身份证 IdCard 字段在 JSON 数据中去掉:

type User struct {
 Name string `json:"name"`
 Age  Int  `json:"int"`
 IdCard string `json:"-"`
}

encoding/json 的源码中和文档中都列举了通过结构体字段标签控制数据 JSON 编码行为的说明:

// Field is ignored by this package.
Field int `json:"-"`

// Field appears in JSON as key "myName".
Field int `json:"myName"`

// Field appears in JSON as key "myName" and
// the field is omitted from the object if its value is empty,
// as defined above.
Field int `json:"myName,omitempty"`

// Field appears in JSON as key "Field" (the default), but
// the field is skipped if empty.
// Note the leading comma.
Field int `json:",omitempty"`

omitempty 这个是字段的数据为空时,在 JSON 中省略这个字段。为的是节省数据空间, Protobuf 编译器生成的结构体代码中每个字段标签中都有 omitempty 。但是在 Api 开发中这个不常用,因为字段不固定对前端很不友好。

对 Protobuf 不了解的可以看我之前写的文章《 Protobuf语言指南 》。

结构体字段标签的 json 注解中都不加 omitempty 后还遇到一种情况,就是数据类型为切片的字段在数据为空的时候会被 JSON 编码为 null 而不是 [] 。这个前端经常会问我没数据的时候能不能不要返回 null ,每回还要多写一个判断。我的说辞都是不能,其实规范点讲是应该返回 [] 的知识我是我自己没找到到解决方法。作为一个在写代码上有强迫症的人,这个问题还是想搞明白的,好在有一天在 StackOverflow 上看到一个答案,才发现是编码的疏忽导致的。

解决空切片在JSON里被编码成null

因为切片的零值为 nil ,无指向内存的地址,所以当以这种形式定义 var f []int 初始化 slice 后,在JSON中将其编码为 null ,如果想在 JSON 中将空 slice 编码为 [] 则需用make初始化 slice为其分配内存地址:

运行下面的例子可以看出两点的区别:

package main

import (
 "encoding/json"
 "fmt"
)

type Person struct {
 Friends []string
}

func main() {
 var f1 []string
 f2 := make([]string, 0)

 json1, _ := json.Marshal(Person{f1})
 json2, _ := json.Marshal(Person{f2})

 fmt.Printf("%s\n", json1)
 fmt.Printf("%s\n", json2)
}

输出:

{"Friends":null
}
{"Friends":[]
}

其实导致这个问题的原因是Go的 append 函数(甩锅),我们都知道引用类型的变量定义后如果没初始化他们的值是 nil ,无指向内存的地址,是无法直接使用的。但是 append 函数在给切片追加元素时会判断切片是否已初始化,没有的话会帮其初始化分配底层数组。我的习惯是先声明切片,然后再在下面的循环代码中向切片追加元素。但是如果循环没有执行,比如你从数据库没查出数据,就会导致对应切片字段在无数据时返回的是 nil 然后被 JSON 编码成了 null 。所以这个算是一个经验总结出来的 Tip 吧在写代码时大家一定要注意了。

这就是我在开发时把数据编码成 JSON 格式时遇到的三个问题和相应的解决方法。加上之前写的 解析JSON的文章 ,两个文章加起来差不多就能汇总日常开发中关于 encoding/json 库使用的各种问题了。

总结

以上所述是小编给大家介绍的如何控制Go编码JSON数据时的行为,希望对大家有所帮助!

(0)

相关推荐

  • Go语言对JSON进行编码和解码的方法

    本文实例讲述了Go语言对JSON进行编码和解码的方法.分享给大家供大家参考.具体如下: json已成为不同平台间传送数据的最佳方式,Golang对json的支持非常好,代码如下: 复制代码 代码如下: package main import (     "fmt"     "encoding/json" ) func main() {     // json encode     j1 := make(map[string]interface{})     j1[&

  • 浅谈django model postgres的json字段编码问题

    django model的json字段的编码器不能有效编码诸如uuid,datetime等数据类型,当直接存储此类型的对象到json字段中为抛出编码异常,这时可以通过JSONField字段的encoder参数指定json编码器,这里直接使用django rest framework 的 JSONEncoder from django.contrib.postgres.fields import JSONField from rest_framework.utils.encoders import

  • Go JSON编码与解码的实现

    在开发应用程序时,客户端(前端页面或APP)与服务端交互是在所难免的,在交互过程传递数据时,最通用和流行格式便是JSON,Go语言提供了encoding/json包,用于处理JSON数据的编码与解码. 除了JSON,XML也常用于前后端的数据交互,不过由于简洁性.可读性和流行程度,JSON用得更加广泛. JSON简介 1. 什么是JSON? JSON全称为Javascript Object Notation,一种数据结构化交互的标准协议,易于阅读与编写,所以在数据交互时广泛使用. 2. JSON

  • 如何控制Go编码JSON数据时的行为(问题及解决方案)

    今天来聊一下我在Go中对数据进行 JSON 编码时遇到次数最多的三个问题以及解决方法,大家来看看是不是也为这些问题挠掉了不少头发. 自定义JSON键名 这个问题加到文章里我是有所犹豫的,因为基本上大家都会,不过属于同类问题我还是放进来了,对新接触 Go 的同学更友好些. 我们先从最常见的一个问题说,首先在Go 程序中要将数据编码成JSON 格式时通常我们会先定义结构体类型,将数据存放到结构体变量中. type Address struct { Type string City string Co

  • getJSON调用后台json数据时函数被调用两次的原因猜想

    近期在做前端开发时候使用到getJSON调用后台去json数据,发现后台的函数被调用两次,函数名称为getMessages, 多方调查结合网上兄弟经验发现,只要函数名不以get开头就没这个问题了, 本人大胆猜测,应该是请求返回的时候构造json数据时,调用所有get开头的函数,然后取得返回值然后构造响应. 所以,以get开头的函数做action的函数时,首先响应请求调用了一次,然后构造响应又调用了一次.

  • python json load json 数据后出现乱序的解决方案

    众所周知:python json 可以转换的json字符串,但是在将其转换为字典时,出现了乱序 字典是一个散列结构,亦即他自身根据key进行排序,无法保证顺序 import json jsonstr = '{"username":"string","age":"int","income":"float","createdTime":"date"}'

  • 上传文件返回的json数据会被提示下载问题解决方案

    最近项目中出现上传文件返回的json数据会被提示下载,只有在ie10+中才会出现这个问题.前端使用jQuery的插件ajaxForm提交表单,后台返回的数据格式为json.代码如下: 后端Python: 复制代码 代码如下: def jsonp(func):     """Wraps JSONified output for JSONP requests."""     @wraps(func)     def decorated_functio

  • 详解iOS通过ASIHTTPRequest提交JSON数据

    先验知识--什么是ASIHTTPRequest? 使用iOS SDK中的HTTP网络请求API,相当的复杂,调用很繁琐,ASIHTTPRequest就是一个对CFNetwork API进行了封装,并且使用起来非常简单的一套API,用Objective-C编写,可以很好的应用在Mac OS X系统和iOS平台的应用程序中.ASIHTTPRequest适用于基本的HTTP请求,和基于REST的服务之间的交互. 上传JSON格式数据 首先给出主功能代码段,然后对代码进行详细解析: NSDictiona

  • 解决spring mvc 返回json数据到ajax报错parseerror问题

    最近使用ajax接收spring mvc传过来的json数据时总是出现parseerror的错误,错误源码如下: 前端: $.ajax({ type: 'POST', url: "groupFunctionEdit", dataType: 'json', contentType: "application/json", data: JSON.stringify(functiondata), success: function(data){ alert('数据加载成功

  • jQuery使用ajax方法解析返回的json数据功能示例

    本文实例讲述了jQuery使用ajax方法解析返回的json数据功能.分享给大家供大家参考,具体如下: 最近在用jQuery的ajax方法传递接收json数据时发现一个问题,那就是返回的data数据,有时候可以直接作为json数据使用,可有时候又不行.查了些资料,解释如下: $.ajax({ url: ajaxurl, type: "POST", success: function(data){ //假设返回的json数据里有status及info2个属性 //有时候可以直接ajaxo

  • android之json数据过长打印不全问题的解决

    在logcat中打后台打印json数据时,json数据太过于长就会有打印不全的问题,这里记录一个办法就是分段打印: // 采用分段打印 四千字符分一段 if (response.length() > 4000) { for (int i = 0; i < response.length(); i += 4000) { if (i + 4000 < response.length()) { Log.i("第" + i + "数据", response

  • Java解析JSON数据时报错问题解决方案

    一.问题由来 测试人员最近在测试时,后台日志一直抱错,大致意思是JSON数据解析错误,错误信息如下: 二.问题分析 去查看代码时,发现异常信息是这里抛出来的,解析时使用的是json-lib这个包中的方法: 然后进一步排查错误,将里面的测试数据复制出来查看能否正常解析,如下: 结果还是不行,进一步排查发现问题,去掉最后一个解析的字段remarks后,解析正常: 由此找到问题的原因,是由于JSON数据中包含回车符\n,数据如下: 三.解决方案 原因找到后,就很好对症下药,针对上面的问题,在解析JSO

  • SpringBoot如何接收前端传来的json数据

    目录 SpringBoot接收前端传来的json数据 使用POJO 使用Map接收 使用POJO和Map接收的比较 PO和VO相结合 SpringBoot接收json入参 总结 SpringBoot接收前端传来的json数据 当前端传来json数据时,后端有两种方式可以接收:使用POJO和Map进行接收. 前端json数据: {     id:121,      //数字     name:"lhj",     //字符串     scoreList:[     //对象       

随机推荐