问答

请问下 golang json 化的问题,重写 UnmarshalJSON 后取不到值了

作者:admin 2021-06-07 我要评论

关于 json 化,定义了 IdArr 的结构体,主要目的是为了接收前端字符数组,自动转换为数值型; 在嵌套到 A 中时,可以取到 ids 的值,more 的值永远为空. package maini...

在说正事之前,我要推荐一个福利:你还在原价购买阿里云、腾讯云、华为云服务器吗?那太亏啦!来这里,新购、升级、续费都打折,能够为您省60%的钱呢!2核4G企业级云服务器低至69元/年,点击进去看看吧>>>)

关于 json 化,定义了 IdArr 的结构体,主要目的是为了接收前端字符数组,自动转换为数值型; 在嵌套到 A 中时,可以取到 ids 的值,more 的值永远为空.

package main

import (
    "encoding/json"
    "fmt"
    "strconv"
)

type IdArr struct {
    Ids []uint64 `json:"ids"`
}

type A struct {
    IdArr
    More string `json:"more"`
}

func (s *IdArr) UnmarshalJSON(data []byte) error {
    type Temp IdArr
    t := struct {
        Ids []string `json:"ids"`
        *Temp
    }{
        Temp: (*Temp)(s),
    }
    if err := json.Unmarshal(data, &t); err != nil {
        return err
    }
    for _, id := range t.Ids {
        uId, err := strconv.ParseInt(id, 10, 64)
        if err != nil {
            return err
        }
        s.Ids = append(s.Ids, uint64(uId))
    }
    return nil
}

func main() {

    d := `
        {"ids":["1213334"], "more": "text"}
                        `
    a := &A{}
    json.Unmarshal([]byte(d), &a)

    fmt.Printf("%+v", a)
}

输出

&{IdArr:{Ids:[1213334]} More:}

https://play.golang.org/p/mjR...

卡了半天了,请问什么原因呢?

###

主要原因还是没有理解golang里面的结构体匿名嵌套和继承的问题:
当IdArr匿名嵌套在A结构体内部时,IdArr的接口会被A继承
所以当你对IdArr进行重写方法UnmarshalJSON时,A就继承了UnmarshalJSON方法
所以在调用json.Unmarshal([]byte(d), &a)时,直接调用了A的UnmarshalJSON,其实就是新定义的UnmarshalJSON方法,
其入参是(转换为string后):{"ids":["1213334"], "more": "text"}
而你的UnmarshalJSON方法其实都是在处理IdArr结构,从而more这个定义被忽略了,导致解析不出来。

修改方法:
1、按照eudore给的最小粒度重写方法来做
2、不要重写IdArr的UnmarshalJSON,而是直接重写A的UnmarshalJSON方法:
(将如下方法替换掉你的UnmarshalJSON)

func (s *A) UnmarshalJSON(data []byte) error {
    t := struct {
        Ids []string `json:"ids"`
        More string `json:"more"`
    }{}

    if err := json.Unmarshal(data, &t); err != nil {
        return err
    }
    for _, id := range t.Ids {
        uId, err := strconv.ParseInt(id, 10, 64)
        if err != nil {
            return err
        }
        s.Ids = append(s.Ids, uint64(uId))
    }
    s.More = t.More
    return nil
}

3、修改结构体的定义,不要匿名嵌套,就不会有继承的问题:
假如你的入参只能是{"ids":["1213334"], "more": "text"}这种结构的话,这种就你的初衷不符合,入参也同样的要改

补充一下,你在UnmarshalJSON中定义的Temp毫无意义。

###

问题在于给定的json串结构与定义的go结构不一样。即,IdArrmore 并不在同一层,moreids键是在同一层,如此这般即可

    t := struct {
        More string   `json:"more"`
        Ids  []string `json:"ids"`
    }{}
    json.Unmarshal([]byte(d), &t)
    fmt.Printf("%+v", t)
    // {More:text Ids:[1213334]}
###

最小粒度重写就好了,瞎组合看了脑子都晕,more是A的属性,而重写IdArr反序列化是没法处理more属性并传递到IdArr外面的A。

package main

import (
    "encoding/json"
    "fmt"
    "strconv"
)

type A struct {
    IdArr
    More string `json:"more"`
}
type IdArr struct {
    Ids []ids `json:"ids"`
}
type ids uint64

func (s *ids) UnmarshalJSON(data []byte) error {
    id := string(data)
    if len(id) > 2 && id[0] == '"' && id[len(id)-1] == '"' {
        id = id[1 : len(id)-1]
    }

    uId, err := strconv.ParseInt(id, 10, 64)
    if err != nil {
        return err
    }
    *s = ids(uId)
    return nil
}

func main() {
    d := `
        {"ids":["1213334"], "more": "text"}
                        `
    a := &A{}
    json.Unmarshal([]byte(d), &a)

    fmt.Printf("%+v", a)
}

版权声明:本文转载自网络,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。本站转载出于传播更多优秀技术知识之目的,如有侵权请联系QQ/微信:153890879删除

相关文章
  • 请问下 golang json 化的问题,重写 Unm

    请问下 golang json 化的问题,重写 Unm

  • C语言中,空字符串中系统会自动添加\0

    C语言中,空字符串中系统会自动添加\0

  • Thinkphp-5 Cache::tag paginate返回的

    Thinkphp-5 Cache::tag paginate返回的

  • JavaScript中 bind函数 返回的 绑定函

    JavaScript中 bind函数 返回的 绑定函

腾讯云代理商
海外云服务器