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

Последнее обновление: 21.02.2022

Большим преимуществом 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}");
Помощь сайту
Юмани:
410011174743222
Перевод на карту
Номер карты:
4048415020898850