首页 >后端开发 >Golang >如何将 json 解组为结构体而不映射所有元素,同时保留所有信息

如何将 json 解组为结构体而不映射所有元素,同时保留所有信息

王林
王林转载
2024-02-05 21:36:04466浏览

如何将 json 解组为结构体而不映射所有元素,同时保留所有信息

问题内容

我从队列中收到一个 JSON 对象。我想处理该对象的部分内容并在发送到另一个队列之前更新它。

如果我有一个输入

{
  "one": "data1",
  "two": "data2",
  "three": "data3",
  ...
}

我想采用 onetwo 并在添加新输出并发送类似消息之前计算一些信息

{
  "one": "data1UPDATED",
  "two": "data2",
  "three": "data3",
  ...
  "extra_info": "extra"
}

为了能够更好地处理数据,我想将 JSON 解组为类似的结构

type Message struct {
 One string `json:one`
 Two string `json:two`
}

但不想映射所有字段,因为其中许多字段与此应用程序无关,而其他字段可能会更改。

我尝试过的事情

我尝试将所有字段映射到 json.RawMessage

type Message struct {
 One string `json:"one"`
 Two string `json:"two"`
 ExtraFields json.RawMessage `json:"-"`
}

但是,在编组结构时,不包括 ExtraFields 中包含的字段。

{
  "one": "data1",
  "two": "data2"
}

我还尝试编写一种自定义解组,将完整消息存储在一个结构中,其中有用消息存储在一个元素中,其他信息存储在另一个字段中

type TmpMessage struct{
  Msg Message
  Others json.RawMessage
}

但这变得非常混乱,我希望有一个更简洁的解决方案。

有没有办法做到这一点,或者是不解组到结构并仅使用原始 map[string] 接口{}的唯一选择?


正确答案


您可以编写一个自定义解组器,例如:

https://www.php.cn/link/a73d9b34d6f7c322fa3e34c633b1297d

func (m *Message) UnmarshalJSON(data []byte) error {
  tmp := make(map[string]interface{})
  if err := json.Unmarshal(data, &tmp); err != nil {
    return err
  }
  t := reflect.TypeOf(*m)
  for i := 0; i < t.NumField(); i++ {
    f := t.Field(i)
    if f.Name != "ExtraFields" {
      jsonTag := f.Tag.Get("json")
      // this part should probably by a type switch...
      reflect.ValueOf(m).Elem().Field(i).SetString(tmp[jsonTag].(string))
      delete(tmp, jsonTag)
    }
  }
  var err error
  m.ExtraFields, err = json.Marshal(tmp)
  return err
}

以上是如何将 json 解组为结构体而不映射所有元素,同时保留所有信息的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文转载于:stackoverflow.com。如有侵权,请联系admin@php.cn删除