在C#的LINQ to XML技术体系中,XElement类提供了操作XML元素的丰富方法,其中Elements方法是用于查找当前元素直接子元素的核心接口,它只会返回当前元素下一层级的直接子元素,不会递归遍历更深层级的后代元素。

Elements方法的基本定义
Elements方法是XElement类的实例方法,有两种常用的重载形式:
- 无参数重载:返回当前XElement的所有直接子元素,返回类型为
IEnumerable<XElement> - 带XName参数重载:返回当前XElement中名称匹配指定XName的直接子元素,返回类型同样为
IEnumerable<XElement>
基础使用示例
首先构造一个简单的XML文档作为测试数据,后续示例都基于该文档展开:
<root>
<user>
<name>张三</name>
<age>25</age>
<address>
<city>北京</city>
</address>
</user>
<user>
<name>李四</name>
<age>30</age>
</user>
</root>
查找所有直接子元素
下面的代码演示如何使用无参数Elements方法获取根元素的所有直接子元素:
using System;
using System.Linq;
using System.Xml.Linq;
class Program
{
static void Main()
{
// 加载XML字符串
XElement root = XElement.Parse(@"<root>
<user>
<name>张三</name>
<age>25</age>
<address>
<city>北京</city>
</address>
</user>
<user>
<name>李四</name>
<age>30</age>
</user>
</root>");
// 获取根元素的所有直接子元素
var directChildren = root.Elements();
foreach (var child in directChildren)
{
Console.WriteLine($"子元素名称:{child.Name},内容:{child.ToString()}");
}
}
}
运行上述代码后,会输出两个user元素,因为user是root的直接子元素,而user内部的name、age等元素属于后代元素,不会被Elements方法返回。
查找指定名称的直接子元素
如果只需要获取特定名称的直接子元素,可以传入对应的元素名称作为参数:
using System;
using System.Linq;
using System.Xml.Linq;
class Program
{
static void Main()
{
XElement root = XElement.Parse(@"<root>
<user>
<name>张三</name>
<age>25</age>
</user>
<user>
<name>李四</name>
<age>30</age>
</user>
<config>
<timeout>30</timeout>
</config>
</root>");
// 获取所有名称为user的直接子元素
var userElements = root.Elements("user");
Console.WriteLine($"找到{userElements.Count()}个user直接子元素");
foreach (var user in userElements)
{
Console.WriteLine($"user元素内容:{user.ToString()}");
}
}
}
Elements与Descendants的区别
很多开发者容易混淆Elements和Descendants方法,两者的核心差异在于遍历范围:
| 方法 | 遍历范围 | 返回结果 |
|---|---|---|
| Elements | 仅当前元素的下一层级直接子元素 | 不包含更深层级的后代元素 |
| Descendants | 当前元素的所有后代元素,递归遍历所有层级 | 包含所有层级的匹配元素 |
以下示例可以直观展示两者的差异:
using System;
using System.Linq;
using System.Xml.Linq;
class Program
{
static void Main()
{
XElement root = XElement.Parse(@"<root>
<user>
<name>张三</name>
<age>25</age>
</user>
</root>");
Console.WriteLine("使用Elements查找name元素:");
// Elements只会找root的直接子元素,不会找user下的name
var elementsResult = root.Elements("name");
Console.WriteLine($"结果数量:{elementsResult.Count()}");
Console.WriteLine("使用Descendants查找name元素:");
// Descendants会递归找所有后代元素
var descendantsResult = root.Descendants("name");
Console.WriteLine($"结果数量:{descendantsResult.Count()}");
foreach (var name in descendantsResult)
{
Console.WriteLine($"name内容:{name.Value}");
}
}
}
运行后可以看到,Elements方法找不到name元素,而Descendants方法可以找到1个name元素。
实际开发注意事项
- Elements方法返回的是延迟执行的结果,只有在遍历时才会实际执行查询,适合处理大型XML文档
- 如果传入的元素名称不存在,Elements方法会返回空集合,不会抛出异常,使用时无需额外做空判断
- 如果需要同时获取直接子元素和属性,可以结合
Nodes方法使用,Nodes会返回当前元素下的所有子节点,包括元素和属性
注意:当XML元素带有命名空间时,使用Elements方法需要传入带命名空间的XName,否则无法匹配到对应元素。例如如果XML根元素定义了命名空间xmlns="http://test.com",那么查找子元素时需要用root.Elements("{http://test.com}user")的形式。