C#如何将XML节点转换为对象 反序列化单个元素

最常用且推荐的方式是使用XmlSerializer,需将XmlNode用OuterXml转为字符串后通过StringReader反序列化;类结构须与XML严格匹配,并用XmlRoot[XmlAttribute]等特性显式声明映射关系。

在C#中,将XML节点(特别是单个元素)反序列化为对象,最常用且推荐的方式是使用 XmlSerializer,但需注意:它默认反序列化整个XML文档或根元素。若你手上只有一个 XmlNode(比如从 XmlDocument 中提取的子节点),需先将其“包装”成独立的、格式合法的XML片段,再交由 XmlSerializer 处理。

确保类结构与XML元素匹配

反序列化成功的关键是目标类的结构(属性名、特性)必须与XML元素的标签名、属性和嵌套关系一致。建议使用 [XmlElement][XmlAttribute] 等特性显式声明映射关系,避免依赖默认命名规则。

例如,有如下XML片段:


  张三
  30

对应类应定义为:

[XmlRoot("Person")]
public class Person
{
    [XmlAttribute("ID")]
    public string ID { get; set; }

    [XmlElement("Name")]
    public string Name { get; set; }

    [XmlElement("Age")]
    public int Age { get; set; }
}

从XmlNode提取并构造可反序列化的XML字符串

XmlSerializer 不直接接受 XmlNode,但可以将节点导出为字符串(含XML声明或不含),再用 StringReader 包装。关键点是:确保导出的是完整、自包含的XML片段(即该节点自身是合法根)。

  • node.OuterXml 获取包含起始标签、内容和结束标签的完整字符串(推荐)
  • 避免用 node.InnerXml,它不包含外层标签,会导致反序列化失败
  • 若节点来自较大文档,OuterXml 已足够,无需额外包装

使用XmlSerializer完成反序列化

创建 XmlSerializer 实例(类型需与目标类一致),传入 StringReader,调用 Deserialize 即可。

public static T DeserializeNode(XmlNode node)
{
    if (node == null) throw new ArgumentNullException(nameof(node));
    
    var xml = node.OuterXml;
    using var reader = new StringReader(xml);
    var serializer = new XmlSerializer(typeof(T));
    return (T)serializer.Deserialize(reader);
}

// 使用示例:
// XmlNode personNode = doc.SelectSingleNode("//Person");
// Person p = DeserializeNode(personNode);

注意事项与常见问题

反序列化单个节点看似简单,但容易踩坑:

  • XML命名空间(xmlns)存在时,类需用 [XmlRoot(Namespace = "...")] 声明,否则匹配失败
  • 若节点含文本内容(如 100),对应属性类型必须能转换(如 intstring),否则抛异常
  • 空元素()反序列化为 0null,取决于属性是否可空;建议用可空类型(如 int?)并配合 [XmlElement(IsNullable = true)]
  • 不支持直接反序列化纯文本节点或属性混合的复杂结构——此时应考虑 XDocument + LINQ to XML 手动映射