推荐使用XDocument递归遍历XML所有节点和属性:从根元素开始,逐层输出元素名、值及属性,并用Nodes()而非Elements()覆盖文本、注释等全类型节点;也可用Descendants().Attributes()一次性提取全部属性。

在C#中遍历XML文件的所有节点和属性,推荐使用 XDocument(LINQ to XML)——它简洁、易读、功能强大,比传统的 XmlDocument 更现代,也比 XmlReader 更适合全树遍历。
用 XDocument 递归遍历所有节点和属性
核心思路是:加载XML后,从根元素开始,递归访问每个元素及其子节点,并在每层输出当前元素的名称、值、属性列表。
示例代码:
using System;
using System.Xml.Linq;
public static void TraverseAllNodes(string filePath)
{
try
{
var doc = XDocument.Load(filePath);
TraverseElement(doc.Root);
}
catch (Exception ex)
{
Console.WriteLine($"加载或解析失败:{ex.Message}");
}
}
private static void TraverseElement(XElement element)
{
if (element == null) return;
// 输出当前元素信息
Console.WriteLine($"元素: <{element.Name}> | 值: \"{element.Value.Trim()}\"");
// 输出所有属性
if (element.HasAttributes)
{
foreach (var attr in element.Attributes())
{
Console.WriteLine($" 属性: {attr.Name} = \"{attr.Value}\"");
}
}
// 递归遍历子元素(跳过文本/空白节点,只处理元素节点)
foreach (var child in element.Elements())
{
TraverseElement(child);
}
}
区分不同节点类型(元素、文本、注释等)
XNode 是所有XML节点的基类,包括 XElement、XText、XComment、XProcessingInstruction 等。若需完整遍历(含文本节点),应使用 element.Nodes() 而非 element.Elements():
-
element.Elements()→ 只返回子 元素节点 -
element.Nodes()→ 返回所有子节点(元素、文本、注释、CDATA等)
例如处理纯文本内容(如 Hello 中的 "Hello" 和 "World"):
World
private static void TraverseAllNodes(XNode node)
{
switch (node)
{
case XElement el:
Console.WriteLine($"元素: <{el.Name}>");
foreach (var attr in el.Attributes())
Console.WriteLine($" 属性: {attr.Name} = \"{attr.Value}\"");
foreach (var child in el.Nodes())
TraverseAllNodes(child);
break;
case XText txt when !string.IsNullOrWhiteSpace(txt.Value):
Console.WriteLine($"文本: \"{txt.Value.Trim()}\"");
break;
case XComment comment:
Console.WriteLine($"注释: {comment.Value}");
break;
case XProcessingInstruction pi:
Console.WriteLine($"处理指令: {pi.Target} {pi.Data}");
break;
}
}
获取所有属性的一次性列表(无需递归)
如果只需提取全部属性(不管属于哪个元素),可用 LINQ 快速扁平化:
var allAttributes = doc.Descendants()
.Attributes()
.Select(a => new { Name = a.Name, Value = a.Value, Owner = a.Parent?.Name });
foreach (var attr in allAttributes)
{
Console.WriteLine($"{attr.Owner}.{attr.Name} = \"{attr.Value}\"");
}
注意:Descendants() 包含所有后代元素(不含根节点自身属性),如需包含根节点属性,可显式加上 doc.Root?.Attributes() 合并。
注意事项与常见问题
-
编码问题:用
XDocument.Load(filePath)会自动识别BOM或声明中的编码;若手动指定,可用new XmlReaderSettings { DtdProcessing = DtdProcessing.Ignore }防止外部DTD引发异常 -
命名空间:含命名空间的XML需用
XNamespace前缀匹配,否则element.Elements("Item")可能返回空 -
性能敏感场景:超大XML(百MB以上)建议改用
XmlReader流式读取,避免全量加载到内存 -
空值安全:始终检查
element?.Attributes()和element?.Nodes()是否为 null,尤其在深层嵌套或不规范XML中










