最好的LINQ-to-XML查询,根据后代节点的属性选择节点?

时间:2009-04-24 19:30:46

标签: c# xml linq-to-xml

我有以下XML文档:

<?xml version="1.0" encoding="UTF-8"?>
<FamilyTree>
  <Parent name="Ken">
    <Child name="Lorna">
      <Grandchild name="Andrew"/>
      <Grandchild name="Brian"/>
    </Child>
    <Child name="Mike">
      <Grandchild name="Ann"/>
      <Grandchild name="Beth"/>
    </Child>
  </Parent>
  <Parent name="Norma">
    <Child name="Owen">
      <Grandchild name="Charles"/>
    </Child>
    <Child name="Peter">
      <Grandchild name="Charlotte"/>
    </Child>
  </Parent>
  <Parent name="Quinn">
    <Child name="Robert">
      <Grandchild name="Debbie"/>
      <Grandchild name="Eric"/>
    </Child>
    <Child name="Susan">
      <Grandchild name="Frank"/>
    </Child>
  </Parent>
  <Parent name="Tom">
    <Child name="Ursula">
      <Grandchild name="George"/>
      <Grandchild name="Harriet"/>
    </Child>
    <Child name="Victor">
      <Grandchild name="Ian"/>
      <Grandchild name="Juliet"/>
    </Child>
  </Parent>
</FamilyTree>

我正试图选择所有“父母”和一个至少有两个孩子(“孙子”)的孩子。请注意,我正在寻找至少有两个“孙子”的“父母”。

以下LINQ查询有效,但我觉得它不是最优雅的。

IEnumerable<XElement> parents = (from c in familyTreeElement.Descendants("Child")
                                 where c.Elements().Count() > 1
                                 select c.Parent).Distinct();

有更好的方法来指定它吗?

3 个答案:

答案 0 :(得分:5)

嗯......我发现我很难完全理解它:)

通常要查明是否有任何元素,我会使用Any - 但您想查看是否有至少两个元素。我们仍然不需要使用Count - 因为至少有两个元素与跳过一个元素并查看是否还有任何元素相同。所以......

var parents = familyTreeElement.Elements("Parent")
    .Where(parent => parent.Elements("Child").Any(
                     child => child.Elements("Grandchild").Skip(1).Any()));

我觉得这很有效 - 实际上它也没有读太多

对于每个家长,在忽略第一个(大)孩子后,查看是否有任何(大)孩子的

我怀疑使用XPath(根据Marc的回答)是最可读的选项。

答案 1 :(得分:2)

啊编辑(2个大孩子)有帮助;-p

虽然XDocument很有用,但我有时会错过XPath / XQuery。使用XmlDocument,您只需使用doc.DocumentElement.SelectNodes("Parent[Child/Grandchild[2]]")

答案 2 :(得分:0)

我不知道“SQL-like”语法足以保证我以正确的方式编写语法,但你想使用.Any()代替.Count() ,如果您以不同的方式选择,则最后不需要Distinct()。试试这个:

IEnumerable<XElement> parents =
    familyTreeElement.Elements("Parent").Where(
        parent => parent.Elements("Child").Any(
            child => child.Elements().Count() >= 2));

编辑:如果您想确保至少有2个,那么您几乎必须使用.Count()