[英]rust: adding a field to an existing struct with serde_json
我有一个预定义的结构
use serde::{Deserialize, Serialize};
#[derive(Debug, Serialize, Deserialize)]
struct Foo<T>
where T: Serialize
{
foo: T
}
struct Bar{
a: String
}
struct Bar2{
b: String
}
fn main() -> Result<()>
{
let a1 = Bar {a: "something".to_owned(),};
let a2 = Bar {a: "something2".to_owned(),};
let a_vec: Vec<Bar> = vec![a1, a2];
let b = Bar2 {b: "something"}
let b_vec: Vec<Bar2> = vec![b];
//let foo = Foo {foo: vec![a_vec,b_vec]}
}
如何将两个结构都放在Foo
下,或者是否可以先将Bar
序列化为 json 并将Bar2
添加为字符串文字? 结果将是一个json
{"foo": [{"a": "something"}, {"a": "something2"}], "b": "something"}
您可以通过将Foo
和Bar2
存储在另一个结构中并将它们与#[serde(flatten)]
合并在一起来获得此序列化结构。 ( 操场):
use serde::{Deserialize, Serialize};
#[derive(Debug, Serialize, Deserialize)]
struct Foo<T>
where
T: Serialize,
{
foo: T,
}
#[derive(Debug, Serialize)]
struct Bar {
a: String,
}
#[derive(Debug, Serialize)]
struct Bar2 {
b: String,
}
#[derive(Debug, Serialize)]
struct Outer<T: Serialize> {
#[serde(flatten)]
field_1: Foo<T>,
#[serde(flatten)]
field_2: Bar2,
}
fn main() {
let a1 = Bar {
a: "something".to_owned(),
};
let a2 = Bar {
a: "something2".to_owned(),
};
let a_vec: Vec<Bar> = vec![a1, a2];
let b = Bar2 {
b: "something".to_owned(),
};
let o = Outer {
field_1: Foo { foo: a_vec },
field_2: b,
};
println!("{}", serde_json::to_string(&o).unwrap());
}
{"foo":[{"a":"something"},{"a":"something2"}],"b":"something"}
如果改为“不修改结构” ,您的意思是仅序列化Foo
并仅修改T
,那么不可能直接使用serde
获得该输出。 您必须通过序列化为Value
并自己合并它们来执行您提出的方法。
我发现您的代码有两个问题:
Bar
和Bar2
派生Serialize
。a_vec
和b_vec
放入一个向量中,但您不能这样做,因为它们的类型不同。 后一个问题可以通过动态调度来解决。 不是将a_vec
放入Vec
,而是将&a_vec as &dyn Serialize
,即只给它一个对某些可序列化对象的引用。 小麻烦是这不适用于serde::Serialize
因为它不是对象安全的。 这就是erased_serde
的用途:
let foo = Foo::<Vec<&dyn erased_serde::Serialize>> {
foo: vec![&a_vec, &b_vec],
};
println!("{}", serde_json::to_string_pretty(&foo).unwrap());
会正常工作。 但它会输出
{"foo":[[{"a":"something"},{"a":"something2"}],[{"b":"something"}]]}
这不是你想要的?
要获得您想要的输出,即序列化 Foo 并向其添加一个字段,您可以使用#[serde(flatten)]
:
let foo = Foo { foo: a_vec };
#[derive(Serialize)]
struct Merge<T1: Serialize, T2: Serialize> {
#[serde(flatten)]
f1: T1,
#[serde(flatten)]
f2: T2,
}
let out = Merge { f1: &foo, f2: &b };
println!("{}", serde_json::to_string_pretty(&out).unwrap());
序列化的输出Merge
来自T1
和T2
的两个字段。
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.