如何控制Go编码JSON数据时的行为(问题及解决方案)
今天来聊一下我在Go中对数据进行JSON编码时遇到次数最多的三个问题以及解决方法,大家来看看是不是也为这些问题挠掉了不少头发。
自定义JSON键名
这个问题加到文章里我是有所犹豫的,因为基本上大家都会,不过属于同类问题我还是放进来了,对新接触Go的同学更友好些。
我们先从最常见的一个问题说,首先在Go程序中要将数据编码成JSON格式时通常我们会先定义结构体类型,将数据存放到结构体变量中。
typeAddressstruct{ Typestring Citystring Countrystring } typeCreditCardstruct{ FirstNamestring LastNamestring Addresses[]*Address Remarkstring } 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("JSONformat:%s",js)
只有导出的结构体成员才会被编码,这也就是我们为什么选择用大写字母开头的字段名称。在编码时,默认使用结构体字段的名字作为JSON对象中的key,但是一般JSON是给HTTP接口返回数据使用的,在接口的规范里针对数据我们一般都要求返回snakecase风格的字段名。解决这个问题的方法是在结构体声明时在结构体字段标签里可以自定义对应的JSONkey
所以我们把结构体声明改为如下即可:
typeAddressstruct{ Typestring`json:"type"` Citystring`json:"city"` Countrystring`json:"country"` }
编码JSON时忽略掉指定字段
并不是所有数据我们都期望编码到JSON中暴露给外部接口的,所以针对一些敏感的字段我们往往希望将其从编码后的JSON数据中忽略掉。那么上面也说了只有导出的结构体成员才会被编码,有的同学会问我直接用小写的字段名不行吗?可是未导出字段只能在包内访问,像这种携带内部敏感数据的往往都是应用的基础数据,由项目的公共包来提供的。那么怎么既能维持字段的导出性又能让其在JSON数据中被忽略掉呢?还是使用结构体的标签进行注解,比如下面定义的结构体,可以把身份证IdCard字段在JSON数据中去掉:
typeUserstruct{ Namestring`json:"name"` AgeInt`json:"int"` IdCardstring`json:"-"` }
encoding/json的源码中和文档中都列举了通过结构体字段标签控制数据JSON编码行为的说明:
//Fieldisignoredbythispackage. Fieldint`json:"-"` //FieldappearsinJSONaskey"myName". Fieldint`json:"myName"` //FieldappearsinJSONaskey"myName"and //thefieldisomittedfromtheobjectifitsvalueisempty, //asdefinedabove. Fieldint`json:"myName,omitempty"` //FieldappearsinJSONaskey"Field"(thedefault),but //thefieldisskippedifempty. //Notetheleadingcomma. Fieldint`json:",omitempty"`
omitempty这个是字段的数据为空时,在JSON中省略这个字段。为的是节省数据空间,Protobuf编译器生成的结构体代码中每个字段标签中都有omitempty。但是在Api开发中这个不常用,因为字段不固定对前端很不友好。
对Protobuf不了解的可以看我之前写的文章《Protobuf语言指南》。
结构体字段标签的json注解中都不加omitempty后还遇到一种情况,就是数据类型为切片的字段在数据为空的时候会被JSON编码为null而不是[]。这个前端经常会问我没数据的时候能不能不要返回null,每回还要多写一个判断。我的说辞都是不能,其实规范点讲是应该返回[]的知识我是我自己没找到到解决方法。作为一个在写代码上有强迫症的人,这个问题还是想搞明白的,好在有一天在StackOverflow上看到一个答案,才发现是编码的疏忽导致的。
解决空切片在JSON里被编码成null
因为切片的零值为nil,无指向内存的地址,所以当以这种形式定义varf[]int初始化slice后,在JSON中将其编码为null,如果想在JSON中将空slice编码为[]则需用make初始化slice为其分配内存地址:
运行下面的例子可以看出两点的区别:
packagemain import( "encoding/json" "fmt" ) typePersonstruct{ Friends[]string } funcmain(){ varf1[]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数据时的行为,希望对大家有所帮助!
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。