我从 API 接收 JSON,响应可以是 30 种类型之一。每种类型都有一组唯一的字段,但所有响应都有一个字段type
来说明它是哪种类型。
我的方法是使用serde。我为每种响应类型创建一个结构并使它们可解码。一旦我有了它,我该如何选择应该将哪个结构用于新收到的消息?
目前,我已经创建了另一个TypeStruct
只有一个字段的结构type
。我将 JSON 解码为 a TypeStruct
,然后根据类型值为接收到的消息选择适当的结构,然后再次解码消息。
我想摆脱这种解码重复。
您可以使用现有的枚举反序列化。我将给出一个分步示例,将您的格式反序列化为以下枚举:
#[derive(Debug, PartialEq, Eq, Deserialize)]
enum MyType {
A {gar: ()},
B {test: i32},
C {blub: String},
}
从一个示例 json 字符串开始:
let json = r#"{"type": "B", "test": 42}"#;
把它变成一个Value
枚举
let mut json: serde_json::Value = serde_json::from_str(json).unwrap();
挖出type
田野
let type_ = {
let obj = json.as_object_mut().expect("object");
let type_ = obj.remove("type").expect("`type` field");
if let serde_json::Value::String(s) = type_ {
s
} else {
panic!("type field not a string");
}
};
创建“正确的”枚举 json。具有单个字段的结构,其中字段的名称是枚举变量,字段的值是变量值
let mut enum_obj = std::collections::BTreeMap::new();
enum_obj.insert(type_, json);
let json = serde_json::Value::Object(enum_obj);
使用生成的 json 反序列化器将 json 转换为枚举的值
let obj: MyType = serde_json::from_value(json).unwrap();