最常用且推荐的方式是使用XmlSerializer,需将XmlNode用OuterXml转为字符串后通过StringReader反序列化;类结构须与XML严格匹配,并用XmlRoot[XmlAttribute]等特性显式声明映射关系。
在C#中,将XML节点(特别是单个元素)反序列化为对象,最常用且推荐的方式是使用 XmlSerializer,但需注意:它默认反序列化整个
XML文档或根元素。若你手上只有一个 XmlNode(比如从 XmlDocument 中提取的子节点),需先将其“包装”成独立的、格式合法的XML片段,再交由 XmlSerializer 处理。
反序列化成功的关键是目标类的结构(属性名、特性)必须与XML元素的标签名、属性和嵌套关系一致。建议使用 [XmlElement]、[XmlAttribute] 等特性显式声明映射关系,避免依赖默认命名规则。
例如,有如下XML片段:
对应类应定义为:
[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; }
}
XmlSerializer 不直接接受 XmlNode,但可以将节点导出为字符串(含XML声明或不含),再用 StringReader 包装。关键点是:确保导出的是完整、自包含的XML片段(即该节点自身是合法根)。
node.OuterXml 获取包含起始标签、内容和结束标签的完整字符串(推荐)node.InnerXml,它不包含外层标签,会导致反序列化失败OuterXml 已足够,无需额外包装创建 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);
反序列化单个节点看似简单,但容易踩坑:
xmlns)存在时,类需用 [XmlRoot(Namespace = "...")] 声明,否则匹配失败100 ),对应属性类型必须能转换(如 int、string),否则抛异常)反序列化为 0 或 null,取决于属性是否可空;建议用可空类型(如 int?)并配合 [XmlElement(IsNullable = true)]
XDocument + LINQ to XML 手动映射