用serde_json解析时是否可以展平子对象字段?

问题描述 投票:5回答:1

#[serde(rename)]似乎是正确的选择,但文档没有说明是否可能或如何做到这一点。

这个JSON对象:

{
   "name" : "myobject"
   "info" : 
   {
      "counter" : "3"
      "foo" : "bar"
   }
}

相应的平面Rust结构应该是:

#[derive(Deserialize)]
struct Object {
    name: String,
    #[serde(rename="info.counter")] // wrong syntax here !!
    count: i32,
    #[serde(rename="info::foo")] // neither this works
    foo: String,
}
json rust serde
1个回答
5
投票

使用属性没有内置的方法来执行此操作,但是您可以为Deserialize类型编写自己的Object impl,它首先反序列化为某些中间帮助器表示,然后将数据重新排列为所需的结构。

#[macro_use]
extern crate serde_derive;

extern crate serde;
extern crate serde_json;

use serde::{Deserialize, Deserializer};

#[derive(Debug)]
struct Object {
    name: String,
    count: i32,
    foo: String,
}

impl<'de> Deserialize<'de> for Object {
    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
        where D: Deserializer<'de>
    {
        #[derive(Deserialize)]
        struct Outer {
            name: String,
            info: Inner,
        }

        #[derive(Deserialize)]
        struct Inner {
            count: i32,
            foo: String,
        }

        let helper = Outer::deserialize(deserializer)?;
        Ok(Object {
            name: helper.name,
            count: helper.info.count,
            foo: helper.info.foo,
        })
    }
}

fn main() {
    let j = r#"{
                 "name": "myobject",
                 "info": {
                   "count": 3,
                   "foo": "bar"
                 }
               }"#;

    println!("{:#?}", serde_json::from_str::<Object>(j).unwrap());
}

输出是:

Object {
    name: "myobject",
    count: 3,
    foo: "bar"
}

有三个实质上不同的地方出现无关紧要的嵌套:

  1. 与其他领域相邻
  2. 本身就处于顶层
  3. 本身低于顶级水平

这三者都需要不同的方法。在这个问题中观察到#1。

要解决#2或#3,请参阅Is there a way to omit wrapper/root objects when deserializing objects with Serde?

© www.soinside.com 2019 - 2024. All rights reserved.