[英]Is it possible to flatten sub-object fields while parsing with serde_json?
#[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,
}
没有使用属性执行此操作的内置方法,但是您可以为您的Object
类型编写自己的Deserialize
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,请参阅使用 Serde 反序列化对象时是否有办法省略包装器/根对象?
现在你可以#[serde(flatten)]
如下:
#[derive(Serialize, Deserialize)]
struct Pagination {
limit: u64,
offset: u64,
total: u64,
}
#[derive(Serialize, Deserialize)]
struct Users {
users: Vec<User>,
#[serde(flatten)]
pagination: Pagination,
}
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.