LINQ to XML optional element query

itsmatt picture itsmatt · Nov 10, 2008 · Viewed 7.5k times · Source

I'm working with an existing XML document which has a structure (in part) like so:

<Group>
    <Entry>
        <Name> Bob </Name>
        <ID> 1 </ID>
    </Entry>
    <Entry>
        <Name> Larry </Name>
    </Entry>
</Group>

I'm using LINQ to XML to query the XDocument to retrieve all these entries as follows:

var items = from g in xDocument.Root.Descendants("Group").Elements("Entry")
    select new
    {
        name = (string)g.element("Name").Value,
        id = g.Elements("ID").Count() > 0 ? (string)g.Element("ID").Value : "none"
    };

The "ID" elements aren't always there and so my solution to this was the Count() jazz above. But I'm wondering if someone has a better way to do this. I'm still getting comfortable with this new stuff and I suspect that there may be a better way to do this than how I'm currently doing it.

Is there a better/more preferred way to do what I want?

Answer

Jacob Carpenter picture Jacob Carpenter · Nov 10, 2008

XElement actually has interesting explicit conversion operators that do the right thing in this case.

So, you rarely actually need to access the .Value property.

This is all you need for your projection:

var items =
    from g in xDocument.Root.Descendants("Group").Elements("Entry")
    select new
    {
        name = (string) g.Element("Name"),
        id = (string) g.Element("ID") ?? "none",
    };

And if you'd prefer to use the value of ID as an integer in your anonymous type:

var items =
    from g in xDocument.Root.Descendants("Group").Elements("Entry")
    select new
    {
        name = (string) g.Element("Name"),
        id = (int?) g.Element("ID"),
    };