Go:使用json时的陷阱

本文是关于使用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]

6、使用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))
}

输出结果为:

{}

7、使用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"}

8、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类型的字段有效。对于任何其他类型都没有效果。

9、将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

10、自定义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
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 227,837评论 6 531
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 98,196评论 3 414
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 175,688评论 0 373
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 62,654评论 1 309
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 71,456评论 6 406
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 54,955评论 1 321
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 43,044评论 3 440
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 42,195评论 0 287
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 48,725评论 1 333
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 40,608评论 3 354
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 42,802评论 1 369
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 38,318评论 5 358
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 44,048评论 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 34,422评论 0 26
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 35,673评论 1 281
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 51,424评论 3 390
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 47,762评论 2 372

推荐阅读更多精彩内容

  • Go中的标准库encoding/json提供JSON格式的序列化和反序列化功能. 序列化struct为JSON P...
    asdzxc阅读 311评论 0 0
  • 标准库中的encoding/xml包提供了XML格式的序列化功能. 将XML解析为struct 解析XML与解析J...
    asdzxc阅读 248评论 0 0
  • 以下内容是我在学习和研究Go时,对Go的特性、重点和注意事项的提取、精练和总结,还有一些学习笔记(注:部分笔记是摘...
    科研者阅读 624评论 0 1
  • 很多程序都需要处理或者发布数据,不管这个程序是要使用数据库,进行网络调用,还是与分布式系统打交道。如果程序需要处理...
    Go语言由浅入深阅读 1,260评论 0 1
  • 原文地址:https://www.liwenzhou.com/posts/Go/json_tricks_in_go...
    李小斌_2018阅读 284评论 0 2