Linq XML запрашивает потомков родителя с указанным значением c в атрибуте - PullRequest
2 голосов
/ 08 января 2020

Мне нужна помощь с LINQ для XML. У меня есть приведенный ниже XML;

<GcctTestData>
  <TestDriveRequest Record="1">
    <element name="Time Zone">(GMT+05:30)</element>
    <element name="Requested Dealer">Concorde</element>
    <element name="Model Year">1999</element>
    <element name="Make">Tata</element>
    <element name="Model/Vehicle Line">Indica</element>
  </TestDriveRequest>
  <TestDriveRequest Record="2">
    <element name="Time Zone">(GMT+05:30)</element>
    <element name="Requested Dealer">Kun"</element>
    <element name="Model Year">2020"</element>
    <element name="Make">BMW"</element>
    <element name="Model/Vehicle Line">3-series</element>
  </TestDriveRequest>
  <TestDriveRequest Record="3">
    <element name="Time Zone">(GMT+05:30)</element>
    <element name="Requested Dealer">KUN Hyundai</element>
    <element name="Model Year">2001</element>
    <element name="Make">Hyundai</element>
    <element name="Model/Vehicle Line">Verna</element>
  </TestDriveRequest>
</GcctTestData>

Я попробовал это, как показано ниже:

IEnumerable<XElement> xElements =
                from element in root.Elements("TestDriveRequest")
                where element.Attribute("Record").Value == "1"
                select element;

            foreach (XElement el in xElements.Descendants().Where(p => !p.HasElements))
            {
                int keyInt = 0;
                string keyName = el.Attribute("name").Value;

                while (keyValuePairs.ContainsKey(keyName))
                {
                    keyName = $"{el.Attribute("name").Value}_{keyInt++}";
                }
                keyValuePairs.Add(keyName, el.Value);
            }                

Мне нужно получить значение атрибута элемента name элемент для того родителя, который имеет значение записи 1

, но запрос linq не получает его ...

После реализации предложения @ anu-viswan Я сталкиваюсь со следующей проблемой enter image description here

Ответы [ 2 ]

2 голосов
/ 08 января 2020

Вам нужно использовать root.Descendants вместо root.Elements.

Например,

IEnumerable<XElement> xElements =   from element in root.Descendants("TestDriveRequest")
                where element.Attribute("Record").Value == "1"
                select element;

XContainer.Elements

Возвращает отфильтрованную коллекцию дочерних элементов этого элемента или документа в порядке документа. В коллекцию включаются только элементы с соответствующим именем XName.

XContainer.Descendants

Возвращает отфильтрованную коллекцию элементов-потомков для этого документ или элемент, в порядке документа. Только элементы, имеющие совпадающее имя XName, включены в коллекцию Output

Полный код

IEnumerable<XElement> xElements =   from element in root.Descendants("TestDriveRequest")
            where element.Attribute("Record").Value == "1"
            select element;

foreach (XElement el in xElements.Descendants().Where(p => !p.HasElements))
{
   int keyInt = 0;
   string keyName = el.Attribute("name").Value;

   while (keyValuePairs.ContainsKey(keyName))
   {
     keyName = $"{el.Attribute("name").Value}_{keyInt++}";
   }
   keyValuePairs.Add(keyName, el.Value);
}  

Пример вывода

enter image description here

0 голосов
/ 08 января 2020

Вот как бы я создал словарь:

using System;
using System.Collections.Generic;
using System.Collections;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;
namespace AscendingSequences
{
    class AscendingSequences
    {
        const string FILENAME = @"c:\temp\test.xml";
        public static void Main(string[] args)
        {
            XDocument doc = XDocument.Load(FILENAME);

            Dictionary<int, Dictionary<string, string>> dict = doc.Descendants("TestDriveRequest")
                .GroupBy(x => (int)x.Attribute("Record"), y => y.Elements("element")
                    .GroupBy(a => (string)a.Attribute("name"), b => (string)b)
                    .ToDictionary(a => a.Key, b => b.FirstOrDefault()))
                    .ToDictionary(x => x.Key, y => y.FirstOrDefault());
        }
    }
}
...