Go 使用json时的陷阱
Posted 技术能量站
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Go 使用json时的陷阱相关的知识,希望对你有一定的参考价值。
目录
4. 整数、time.Time和net.IP值可以作为map的key
7. 使用omitempty在time.Time的零值也会失效
9. 将json的number反序列化到interface会转为float64类型
10. 自定义MarshalJSON()方法返回的字符串值必须加引号
本文是关于使用Go的encoding/json包时需要注意的一些会让人迷惑的内容。如果您仔细地阅读官方包文档,就会发现其中有许多内容都提到了,所以从理论上讲,这些内容应该不会让您感到惊讶。但其中有一些根本没有在文档中提到,或者至少没有明确指出-值得注意!
1. json序列化map的内容是按照字母排序的
当将一个map编码为json,其内容将根据键值以字母顺序排列,例如:
func main()
m := map[string]int
"z": 123,
"0": 123,
"a": 123,
"_": 123,
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
结果:
"0":123,"_":123,"a":123,"z":123
2. byte切片将编码为base64字符串
当将任何[]byte切片编码为JSON时,它们将被转换为base64编码的字符串。base64字符串使用填充和标准编码字符,如RFC4648中定义的那样。例如,下面的map:
func main()
m := map[string][]byte
"foo": []byte("bar baz"),
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
结果为:
"foo":"YmFyIGJheg=="
3. Nil和空切片编码结果不一样
Go中的空切片将被编码为null JSON值。相反,空的(但不是nil的)切片将被编码为空JSON数组。例如:
func main()
var nilSlice []string
emptySlice := []string
m := map[string][]string
"nilSlice": nilSlice,
"emptySlice": emptySlice,
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
编码结果:
"emptySlice":[],"nilSlice":null
4. 整数、time.Time和net.IP值可以作为map的key
map以整数值为key可以被序列化为json。这些整数将被自动转换为JSON中的字符串(因为JSON对象中的键必须总是字符串)。例如:
func main()
m := map[int]string
123: "foo",
456_000: "bar",
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
"123":"foo","456000":"bar"
此外,Go还允许实现了encoding.TextMarshaler接口的键对map序列化。这意味着你可以直接使用time.Time和net.IP值作为map的key。例如:
func main()
t1 := time.Now()
t2 := t1.Add(24 * time.Hour)
m := map[time.Time]string
t1: "foo",
t2: "bar",
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
"2021-09-19T07:26:03.938939+08:00":"foo","2021-09-20T07:26:03.938939+08:00":"bar"
注意,如果使用其他类型作为map的键进行编码将会得到一个json.UnsupportedTypeError错误。
5. 字符串中的尖括号和&符号被转义
如果一个字符串包含尖括号<>,在JSON中将转义为\\u003c和\\u003e。同样,&字符将转义为\\u0026。这是为了防止某些web浏览器不小心将JSON解释为html。例如:
func main()
m := []string
"<foo>",
"bar & baz",
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
["\\u003cfoo\\u003e","bar \\u0026 baz"]
如果你需要将特殊符号保持原来格式编码,可以使用json.Encoder对象并调用setEscapeHTML(false)即可。
6. 浮点数末尾零被删除
当编码一个以0结尾的小数部分的浮点数时,JSON中不会出现任何尾随的0。例如:
func main()
m := []float64
123.0,
456.100,
789.990,
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
[123,456.1,789.99]
7. 使用omitempty在结构体类型时会失效。
omitempty指令从不认为struct类型是空的-即使所有的struct字段都有零值,并且在这些字段上使用了omitempty。它将始终以JSON中的对象形式出现。例如:
func main()
m := struct
Foo struct
Bar string `json:",omitempty"`
`json:",omitempty"`
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
结果:
"Foo":
如果要实现结构体输出空,可以使用指针来定义,omitempty对nil会生效。
func main()
m := struct
Foo *struct
Bar string `json:",omitempty"`
`json:",omitempty"`
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果为:
8. 使用omitempty在time.Time的零值也会失效
在零值时间上使用omitempty。time.Time字段不会在编码的JSON中隐藏。这是因为时间time.Time是一个struct类型,如上所述,omitempty从不将一个结构类型视为空。因此,字符串"0001-01-01 t00:00:00 - 00z "将出现在JSON中(这是在零值time.Time上调用MarshalJSON()方法返回的值。例如:
func main()
m := struct
Foo time.Time `json:",omitempty"`
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
"Foo":"0001-01-01T00:00:00Z"
可以使用引用的指针类型解决
9. string标签强转类型
Go提供了一个字符串结构标记,它强制将单个字段中的数据编码为JSON中的字符串。例如,如果你想强制将一个整数表示为字符串而不是JSON数字,你可以使用string指令,如下所示:
func main()
m := struct
Foo int `json:",string"`
Foo: 123,
marshal, _ := json.Marshal(m)
fmt.Println(string(marshal))
输出结果:
"Foo":"123"
注意,string标记只对包含float、integer或bool类型的字段有效。对于任何其他类型都没有效果。
10. 将json的number反序列化到interface会转为float64类型
当将JSON数字解码为interface类型时,该值将被转为float64类型,即使原始JSON中是整数。如果要保持整数输出可以使用json.Decoder实例并调用UseNumber函数如下所示:
func main()
js := `"foo": 123, "bar": true`
var m map[string]interface
dec := json.NewDecoder(strings.NewReader(js))
dec.UseNumber()
err := dec.Decode(&m)
if err != nil
log.Fatal(err)
i, err := m["foo"].(json.Number).Int64()
if err != nil
log.Fatal(err)
fmt.Printf("foo: %d", i)
输出结果:
foo: 123
11. 自定义MarshalJSON()方法返回的字符串值必须加引号
如果您正在创建一个返回字符串值的自定义MarshalJSON()方法,则必须在返回字符串之前用双引号包装该字符串,否则它将不会被解释为JSON字符串,并将导致运行时错误。例如:
type Age int
func (age Age) MarshalJSON() ([]byte, error)
encodedAge := fmt.Sprintf("%d years", age)
encodedAge = strconv.Quote(encodedAge) // 返回之前用引号将字符串括起来
return []byte(encodedAge), nil
func main()
users := map[string]Age
"alice": 21,
"bob": 84,
js, err := json.Marshal(users)
if err != nil
log.Fatal(err)
fmt.Printf("%s", js)
输出结果:
"alice":"21 years","bob":"84 years"
如果,在上面的代码中,MarshalJSON()的返回值没有使用strconv.Quote,你会得到错误:
2021/09/19 08:04:25 json: error calling MarshalJSON for type main.Age: invalid character 'y' after top-level value
以上是关于Go 使用json时的陷阱的主要内容,如果未能解决你的问题,请参考以下文章
使用 Go 编程语言 66 个陷阱:Golang 开发者的陷阱和常见错误指北