Выборка элементов в LINQ to XML


Большим преимуществом LINQ to XML является то, что эта технология позволяет легко извлекать нужные элементы из документа xml. Например, возьмем xml-файл people.xml, созданный в прошлой теме:

<?xml version="1.0" encoding="utf-8"?>
<people>
<person name="Tom">
<company>Microsoft</company>
<age>37</age>
</person>
<person name="Bob">
<company>Google</company>
<age>41</age>
</person>
</people>
Переберем его элементы и выведем их значения на консоль:

using System.Xml.Linq;

XDocument xdoc = XDocument.Load("people.xml");
// получаем корневой узел
XElement? people = xdoc.Element("people");
if (people is not null)
{
// проходим по всем элементам person
foreach (XElement person in people.Elements("person"))
{

XAttribute? name = person.Attribute("name");
XElement? company = person.Element("company");
XElement? age = person.Element("age");

Console.WriteLine($"Person: {name?.Value}");
Console.WriteLine($"Company: {company?.Value}");
Console.WriteLine($"Age: {age?.Value}");

Console.WriteLine(); // для разделения при выводе на консоль
}
}
И мы получим следующий вывод:

Person: Tom
Company: Microsoft
Age: 37

Person: Bob
Company: Google
Age: 41
Чтобы начать работу с имеющимся xml-файлом, надо сначала загрузить его с помощью статического метода XDocument.Load(), в который передается путь к файлу.

XDocument xdoc = XDocument.Load("people.xml");
Поскольку xml хранит иерархически выстроенные элементы, то и для доступа к элементам надо идти начиная с высшего уровня в этой иерархии и далее вниз. Так, для получения элементов person и доступа к ним надо сначала обратиться к корневому элементу, а через него уже к элементам person:

// получаем корневой узел
XElement? people = xdoc.Element("people");
if (people is not null)
{
// проходим по всем элементам person
foreach (XElement person in people.Elements("person"))
{
Метод Element("имя_элемента") возвращает первый найденный элемент с таким именем. Метод Elements("имя_элемента") возвращает коллекцию одноименных элементов. В данном случае мы получаем коллекцию элементов person и поэтому можем перебрать ее в цикле.

Спускаясь дальше по иерархии вниз, мы можем получить атрибуты или вложенные элементы, например, получение элемента <company>

XElement? company = person.Element("company");
Значение простых элементов, которые содержат один текст, можно получить с помощью свойства Value:

string? companyValue = person.Element("company")?.Value;
Сочетая операторы Linq и LINQ to XML можно довольно просто извлечь из документа данные и затем обработать их. Например:

using System.Xml.Linq;

XDocument xdoc = XDocument.Load("people.xml");

var microsoft = xdoc.Element("people")? // получаем корневой узел people
.Elements("person") // получаем все элементы person
// получаем все person, у которого company = Microsoft
.Where(p => p.Element("company")?.Value == "Microsoft")
.Select(p => new // для каждого объекта создаем анонимный объект
{
name = p.Attribute("name")?.Value,
age = p.Element("age")?.Value,
company = p.Element("company")?.Value
});

if (microsoft is not null)
{
foreach (var person in microsoft)
{
Console.WriteLine($"Name: {person.name} Age: {person.age}");
}
}
В данном случае выбираем все элементы person, у которых вложенный элемент "company" равен "Microsoft". Далее на основе полученной выборке создаем набор анонимных объектов с тремя свойствами. Под вывод также можно было бы создать специально какой-нибудь класс или структуру и использовать их вместо анонимного объекта.

Другой пример - выберем элемент person, в котором атрибут name равен "Tom":

using System.Xml.Linq;

XDocument xdoc = XDocument.Load("people.xml");

var tom = xdoc.Element("people")? // получаем корневой узел people
.Elements("person") // получаем все элементы person
.FirstOrDefault(p => p.Attribute("name")?.Value == "Tom");

var name = tom?.Attribute("name")?.Value;
var age = tom?.Element("age")?.Value;
var company = tom?.Element("company")?.Value;

Console.WriteLine($"Name: {name} Age: {age} Company: {company}");