我有以下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();
有更好的方法来指定它吗?
答案 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()
。