serde-xml-rs 仅支持无命名空间、结构清晰的简单XML,不支持DTD、CDATA、复杂属性及命名空间;需用0.6.x版本并启用serde的derive特性;推荐小中型固定结构场景,复杂需求应选roxmltree或quick-xml。

serde-xml-rs 本身不直接支持 XML 的完整标准(如 DTD、命名空间、处理指令、CDATA 等),但它能可靠处理结构清晰、标签嵌套合理、无命名空间的常规 XML。如果你的 XML 是配置文件、简单 API 响应或内部数据交换格式,它够用;如果要解析带 xmlns、xml:lang 或混合文本/子元素的复杂文档,就得换方案(比如 roxmltree + 手动映射)。
怎么加依赖和基础序列化/反序列化
这是最常卡住的第一步:依赖版本必须对齐,否则编译直接报错。
-
serde必须启用derivefeature,否则#[derive(Serialize, Deserialize)]不生效 -
serde_xml_rs推荐用0.6.x(截至 2025 年底最新稳定版),别用旧版0.5,它和新版serde兼容性差 - 不需要手动引入
xml-rs——它是serde_xml_rs的底层依赖,自动拉取
在 Cargo.toml 中写:
[dependencies]
serde = { version = "1.0", features = ["derive"] }
serde_xml_rs = "0.6"然后就能用 from_str() 和 to_string():
use serde::{Serialize, Deserialize};
use serde_xml_rs::{from_str, to_string};
[derive(Debug, Serialize, Deserialize)]
struct User {
name: String,
age: u8,
}
fn main() -> Result<(), Box> {
let xml = r#"Alice 32 "#;
let user: User = from_str(xml)?;
println!("parsed: {:?}", user);
let out_xml = to_string(&user)?;
println!("{}", out_xml); // 输出格式较简陋(无缩进、单标签不自闭合)
Ok(())}
字段名和 XML 标签名不一致?用 #[serde(rename = "...")]
XML 标签常用 kebab-case(如 )或带属性(如 ),Rust 字段名却只能是 snake_case 或 camelCase。硬凑字段名既难读又易错。
-
#[serde(rename = "first-name")]强制指定 XML 中的标签名 -
#[serde(rename(serialize = "ID", deserialize = "id"))]可分别控制序列化/反序列化时的名称 - 属性要用
#[serde(rename = "$value")](值)或#[serde(rename = "$attr")](需配合#[serde(default)]处理缺失)——但注意:serde_xml_rs对属性支持有限,复杂属性结构建议改用roxmltree
示例:
#[derive(Debug, Serialize, Deserialize)]
struct Item {
#[serde(rename = "item-name")]
name: String,
#[serde(rename = "$attr")]
id: Option, // 注意:此写法仅在部分版本中有效,0.6 更推荐用专用属性结构
} 空标签、缺失字段、混合内容?小心 panic 和默认值
serde_xml_rs 默认要求 XML 中每个字段都存在,且类型严格匹配。一旦遇到 (空标签)、 或字段完全缺失,from_str() 就会返回 Err。
- 用
#[serde(default)]给字段设默认值(如String::new()或Vec::new()) - 用
#[serde(default = "default_age")]指向一个零参函数,适合复杂默认逻辑 - 字段类型用
Option是最安全的做法,能自然容纳缺失或空值 - 不要指望它自动把
和25 当成等价——前者可解析,后者会失败,除非你加default
大文件或流式解析?它不支持,别硬扛
serde_xml_rs 是基于 DOM 的全量加载解析器:整个 XML 字符串先读进内存,再构造成结构体。这意味着:
- 解析 10MB XML 文件 → 内存占用可能翻倍(字符串 + 结构体)
- 没有
from_reader的真正流式支持(虽然有> from_reader函数,但它仍会把整个 reader 读完才开始解析) - 遇到格式错误时,错误位置信息很弱,只报 “invalid value” 之类,没行号
真要处理大 XML 或需要精确错误定位,换成:
-
roxmltree(轻量、无分配、支持 XPath 查询)+ 手动提取字段 -
quick-xml(高性能流式读/写,但需自己写事件循环)
而 serde_xml_rs 的定位很明确:小到中等、结构固定、开发求快不求极致性能的场景。
最后提醒一句:XML 属性、命名空间、注释、CDATA 这几块,serde_xml_rs 要么不支持,要么行为不稳定。别在关键路径上赌它的兼容性——先用 roxmltree 解析出节点树,再按需转结构体,反而更稳。










