Подтвердить что ты не робот

Удалить элементы empty/blank в коллекции узлов XML

У меня есть XML-документ, подобный этому:

<magento_api>
    <data_item>
        <code>400</code>
        <message>Attribute weight is not applicable for product type Configurable Product</message>
    </data_item>
    <data_item>
        <code>400</code>
        <message>Resource data pre-validation error.</message>
    </data_item>
    <data_item>
        <code>1</code>
        <message></message>
    </data_item>
    <data_item>
        <code></code>
        <message>No code was given</message>
    </data_item>
</magento_api>

Я пытаюсь выполнить итерацию каждого node и сделать следующее:

  • Выбросьте любые пустые/пустые элементы.
  • Создайте новый node только с элементами, содержащими значения.
  • Отправьте полученный документ в другую веб-службу.

Часть, с которой я борюсь, состоит в том, как выполнить итерацию через каждый node и проверить каждый элемент для нулевых значений.

Я тестировал этот код на http://rextester.com/runcode, но, похоже, не понял его:

Console.WriteLine("Querying tree loaded with XElement.Load");
Console.WriteLine("----");
XElement doc = XElement.Parse(@"<magento_api>
          <data_item>
            <code>400</code>
            <message>Attribute weight is not applicable for product type Configurable Product</message>
          </data_item>
          <data_item>
            <code>400</code>
            <message>Resource data pre-validation error.</message>
          </data_item>
          <data_item>
            <code>1</code>
            <message></message>
          </data_item>
          <data_item>
            <code></code>
            <message>No code was given</message>
          </data_item>
    </magento_api>");

int counter = 1;
IEnumerable<XNode> nodes =
    from nd in doc.Nodes()
    select nd;
foreach (XNode node in nodes)
{
    Console.WriteLine(counter + "-" + node);
    IEnumerable<XElement> elements =
    from el in node //this is where I've been trying various methods, but no dice.
    select el;
    foreach (XElement e in elements)
    {
           Console.WriteLine(counter + "-" + e.Name + "-" + e.Value + "\r\n");
    }
    counter++;
}

Основываясь на приведенном выше XML-вводе, я надеюсь получить следующий результат:

<magento_api>
    <data_item>
        <code>400</code>
        <message>Attribute weight is not applicable for product type Configurable Product</message>
    </data_item>
    <data_item>
        <code>400</code>
        <message>Resource data pre-validation error.</message>
    </data_item>
    <data_item>
        <code>1</code>
    </data_item>
    <data_item>
        <message>No code was given</message>
    </data_item>
</magento_api>

Я не уверен, что я использую правильные методы для итерации по узлам и элементам.

4b9b3361

Ответ 1

Один одиночный лайнер может выполнять эту работу, не нужно перебирать все элементы. Вот оно:

doc.Descendants().Where(e => string.IsNullOrEmpty(e.Value)).Remove();

тестер

using System;
using System.Collections.Generic;
using System.Linq;
using System.Text;
using System.Xml;
using System.Xml.Linq;

namespace ConsoleApplication1
{
    public class TestRemove
    {
        public static void Main() {
            Console.WriteLine("----OLD TREE STARTS---");
            XElement doc = XElement.Parse(@"<magento_api>
                                              <data_item>
                                                <code>400</code>
                                                <message>Attribute weight is not applicable for product type Configurable Product</message>
                                              </data_item>
                                              <data_item>
                                                <code>400</code>
                                                <message>Resource data pre-validation error.</message>
                                              </data_item>
                                              <data_item>
                                                <code>1</code>
                                                <message></message>
                                              </data_item>
                                              <data_item>
                                                <code></code>
                                                <message>No code was given</message>
                                              </data_item>
                                        </magento_api>");
            Console.Write(doc.ToString());
            Console.WriteLine("");
            Console.WriteLine("----OLD TREE ENDS---");
            Console.WriteLine("");
            doc.Descendants().Where(e => string.IsNullOrEmpty(e.Value)).Remove();
            Console.WriteLine("----NEW TREE STARTS---");
            Console.Write(doc.ToString());
            Console.WriteLine("");
            Console.WriteLine("----NEW TREE ENDS---");
            Console.ReadKey();
        }
    }
}

И он также может быть протестирован здесь

Ответ 2

doc.Descendants().Where(e => string.IsNullOrEmpty(e.Value)).Remove(); 

В этой строке не будут выбрасываться пустые родительские теги, заполненные пустыми тегами для детей. Он просто удалит своих детей, что может или не может быть уместным в вашей ситуации. Это очень простое изменение для этого, вам просто нужно сначала начать с самого низкого уровня. Что-то вроде

foreach(XElement child in doc.Descendants().Reverse())
{
    if(!child.HasElements && string.IsNullOrEmpty(child.Value) && !child.HasAttributes) child.Remove();
}

Спасибо Nyerguds за предложение атрибута.

Ответ 3

В VB в случае, если мне нужно найти его снова:

doc.Descendants().Where(Function(e) String.IsNullOrEmpty(e.Value)).Remove()