Harobed
Harobed

Reputation: 167

How to group XML nodes based on their values in C#

I have my XML as:

<root>
    <element>
        <id>1</id>
        <group>first</group>
    </element>
    <element>
        <id>2</id>
        <group>second</group>
    </element>
    <element>
        <id>3</id>
        <group>first</group>
    </element>
</root> 

Is there anyway we can group the nodes with same values like this:

<root>
    <groups name="first">
      <element>
        <id>1</id>
        <group>first</group>
      </element>
      <element>
        <id>3</id>
        <group>first</group>
    </element>
  </groups>
   <groups name="second"><element>
       <id>2</id>
        <group>second</group>
    </element>
  </groups>
</root>

Is there a way to group it based on same node values?

Upvotes: 1

Views: 4497

Answers (4)

dbasnett
dbasnett

Reputation: 11773

I had to try this in VB. My Group By skills need a lot of practice.

Using this test data

    Dim myXML As XElement
    myXML = <root>
                <element>
                    <id>1</id>
                    <group>first</group>
                </element>
                <element>
                    <id>2</id>
                    <group>second</group>
                </element>
                <element>
                    <id>3</id>
                    <group>first</group>
                </element>
            </root>

this seems to work

    Dim newXML As XElement = <root></root>

    newXML.Add(From el In myXML.Elements
               Order By el.<group>.Value
               Group By gn = el.<group>.Value Into g = Group
               Select New XElement("Groups", New XAttribute("name", gn), g))

newXML =

<root>
  <Groups name="first">
    <element>
      <id>1</id>
      <group>first</group>
    </element>
    <element>
      <id>3</id>
      <group>first</group>
    </element>
  </Groups>
  <Groups name="second">
    <element>
      <id>2</id>
      <group>second</group>
    </element>
  </Groups>
</root>

Similar to other answers.

Upvotes: 0

Jeff Mercado
Jeff Mercado

Reputation: 134801

Group the elements and build up a new document placing each group in a new <groups> element.

var newDoc = new XDocument(
    new XElement("root",
        from e in doc.Descendants("element")
        group e by (string)e.Element("group") into g
        select new XElement("groups",
            new XAttribute("name", g.Key),
            g
        )
    )
);

Upvotes: 0

jdweng
jdweng

Reputation: 34421

I just tested the code below and it matches your results.

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


namespace ConsoleApplication1
{
    class Program
    {
        static void Main(string[] args)
        {
            string xml =
                "<root>" +
                    "<element>" +
                        "<id>1</id>" +
                        "<group>first</group>" +
                    "</element>" +
                    "<element>" +
                        "<id>2</id>" +
                        "<group>second</group>" +
                    "</element>" +
                    "<element>" +
                        "<id>3</id>" +
                        "<group>first</group>" +
                    "</element>" +
                "</root>";

            XDocument doc = XDocument.Parse(xml);

            var groups = doc.Descendants("element")
                .GroupBy(x => (string)x.Element("group"))
                .ToList();


            XElement newXml = new XElement("root");
            foreach(var group in groups)
            {
                newXml.Add(new XElement("groups", new object[] {
                    new XAttribute("name", group.Key),
                        group
                }));
            }

        }
    }
}

Upvotes: 2

Midhun Mundayadan
Midhun Mundayadan

Reputation: 3182

XPath approach

string val= "first";
XmlDocument doc = new XmlDocument();
doc.Load(Server.MapPath("./Xml/YourXML.xml"));
string text = string.Empty;
XmlNodeList xnl = doc.SelectNodes("/root/groups ");
foreach (XmlNode node in xnl)
{
    text = node.Attributes["name"].InnerText;
    if (text == val)
    {
         XmlNodeList xnl = doc.SelectNodes(string.Format("/root/groups [@name='{0}']/element", val));
        foreach (XmlNode node2 in xnl )
        {
            text = text + "<br>" + node2["id"].InnerText;
            text = text + "<br>" + node2["group"].InnerText;
        }
    }
    Response.Write(text);
}

or

var nodes = (from n in xml.Descendants("element").
             Where(r => r.Parent.Attribute("name").Value == "first")
             select new
             {
                  id = (string)n.Element("id").Value,
                  group = (string)n.Element("group").Value
             }).ToList();

Upvotes: 0

Related Questions