我想解析一个可能包含不同数据类型的XML元素。
下面是我能想到的最简单的例子:
use serde_derive::{Deserialize, Serialize};
use serde_xml_rs::{from_str};
const XML: &str = r#"
<element>
foo
</element>
"#;
const XML2: &str = r#"
<element>
123
</element>
"#;
#[derive(Serialize, Deserialize, Debug, PartialEq)]
enum Element {
#[serde(rename="element")]
Text(String),
#[serde(rename="element")]
Number(i32),
}
fn main() {
let el: Element = from_str(&XML).unwrap();
println!("{:?}", &el);
println!("\n\n\n\n");
let el2: Element = from_str(&XML2).unwrap();
println!("{:?}", &el2);
}当前代码将这两个元素分别解析为文本(String),而不是文本(String)和数字(I32)。
如果我颠倒顺序,程序就会崩溃,因为"foo“不能被解析为整数。
#[derive(Serialize, Deserialize, Debug, PartialEq)]
enum Element {
#[serde(rename="element")]
Number(i32),
#[serde(rename="element")]
Text(String),
}我如何正确地实现我的程序?
发布于 2021-10-04 18:42:15
如果我没理解错的话,只要值可以被解析为i32,你就需要Number,否则就需要Text。您可以使用FromStr特征实现这一点。
#[derive(Serialize, serde_with::DeserializeFromStr, Debug, PartialEq)]
enum Element {
#[serde(rename="element")]
Text(String),
#[serde(rename="element")]
Number(i32),
}
impl std::str::FromStr for Element {
type Err = std::convert::Infallible;
fn from_str(s: &str) -> Result<Self, Self::Err> {
if let Ok(i) = s.parse::<i32>() {
Ok(Element::Number(i))
} else {
Ok(Element::Text(s.to_string()))
}
}
}https://stackoverflow.com/questions/69435590
复制相似问题