Удалить все содержимое родительского узла, если какой-либо дочерний узел не имеет значения - PullRequest
0 голосов
/ 20 декабря 2018

Я хочу удалить родительский узел, если какой-либо из его дочерних элементов не имеет значения.
Я пробовал несколько методов, но ничего не работает.Удалите родительский "S_Industry_Code" и все его дочерние узлы, если дочерний узел "E_Code_List_Qualifier" или "E_Industry_Code пуст. Я хочу общий случай, чтобы код можно было использовать и в других пустых родительских или дочерних узлах. Пожалуйста, помогите! Любые предложения?

namespace TEST
{
    class Remove_Empty_Tags
    {    
        public static void Main()
        {    
            XmlDocument doc = new XmlDocument();               
            doc.Load(@"C:\Users\TestDLMS\out\I_636806391809983753.xml");    
            new RemoveNulls().RemoveEmptyNodes(doc);    
            doc.Save(@"C:\users\Desktop\AllNullsRemoved.xml");    
        }    
    }
}   

   class RemoveNulls
   {
        public void RemoveEmptyNodes(XmlDocument doc)
        {
            XmlNodeList nodes = doc.SelectNodes("//node()");
            foreach (XmlNode node in nodes)
                if ((node.Attributes != null && node.Attributes.Count == 0) && (node.ChildNodes != null && node.ChildNodes.Count == 0))
                {
                  //  node.ParentNode.RemoveChild(node); //removes only nodes that are blank
                      node.ParentNode.RemoveAll(); //removes child nodes leaves empty parent 
                  //  node.ParentNode.ParentNode.RemoveAll(); //removes everything
                }
       }
   }
// Content of XML file
<File>
  <T_Requisition_511R Standard="X12">
    <S_Transaction_Set_Header>
      <E_Transaction_Set_Code>511</E_Transaction_Set_Code>       
      <E_Transaction_Set_Number>0001</E_Transaction_Set_Number>
    </S_Transaction_Set_Header>
    <S_Beginning_Segment_for_Material_Management>
      <E_Transaction_Set_Purpose_Code>00</E_Transaction_Set_Purpose_Code>
      <E_Transaction_Type_Code>A0</E_Transaction_Type_Code>
      <E_Date>20181217</E_Date>
      <E_Time>152620</E_Time>
    </S_Beginning_Segment_for_Material_Management>
    <L_Assigned_Number>
      <L_Code_Source_Information>
        <S_Industry_Code>
          <E_Code_List_Qualifier_Code>A9</E_Code_List_Qualifier_Code>
          <E_Industry_Code>    </E_Industry_Code>
        </S_Industry_Code>
        <S_Industry_Code>
          <E_Code_List_Qualifier_Code>79</E_Code_List_Qualifier_Code>
          <E_Industry_Code>03</E_Industry_Code>
        </S_Industry_Code>
        <S_Industry_Code>
          <E_Code_List_Qualifier_Code>80</E_Code_List_Qualifier_Code>
          <E_Industry_Code>   </E_Industry_Code>
        </S_Industry_Code>
      </L_Code_Source_Information>
    </L_Assigned_Number>
  </T_Requisition_511R>
</File>

Ответы [ 2 ]

0 голосов
/ 20 декабря 2018
string xml_string = @"<File>...</File>";
var xml = XElement.Parse(xml_string);
var empty = xml.Descendants()
            .Where(d => d.Elements().Count() == 0 && d.Value.Trim() == "")
            .Select(d => d.Parent);
empty.Remove();
0 голосов
/ 20 декабря 2018

Вы не проверяете случай, когда атрибуты равны нулю.

Это проще для IMO, используя XDocument против XmlDocument.Вот пример:

void Main()
{
    var doc = XDocument.Load(@"D:\file.xml");

    doc =  RemoveNulls.RemoveEmptyNodes(doc);
    doc.Dump();
}

// Define other methods and classes here
class RemoveNulls
{
    public static XDocument RemoveEmptyNodes(XDocument doc)
    {
        foreach (XElement element in doc.Descendants().ToList()){

            if(!element.HasAttributes && String.IsNullOrWhiteSpace(element.Value) && !element.HasElements)
                element.Remove();
        }
        return doc;     
    }
}
...