仅当子元素具有特定数量的元素时才选择父元素

时间:2013-01-22 11:53:53

标签: c# xml xpath linq-to-xml

我正在尝试获取<Opening>的{​​{1}} <PlanarGeometry><Polyloop>&gt;的所有CartesianPoint代码4。

Xml标签表面是另一个的孩子。

<Surface id="su-137" surfaceType="InteriorWall" constructionIdRef="ASHIW23" xmlns="http://www.gbxml.org/schema">
 <Name>W-106-114-I-W-137</Name>
 <Opening id="su-137-op-1" openingType="NonSlidingDoor" constructionIdRef="MDOOR">
 <Name>W-106-114-I-W-137-D-1</Name>
   <PlanarGeometry>
      <PolyLoop> 
         <CartesianPoint><Coordinate>55.570238</Coordinate><Coordinate>92.571596</Coordinate>
         <Coordinate>0.000000</Coordinate></CartesianPoint><CartesianPoint>         <Coordinate>55.570238</Coordinate><Coordinate>92.571596</Coordinate><Coordinate>6.666667</Coordinate>     
         </CartesianPoint>
         <CartesianPoint>
         <Coordinate>55.570238</Coordinate><Coordinate>95.571596</Coordinate><Coordinate>6.666667</Coordinate></CartesianPoint>
         <CartesianPoint>
         <Coordinate>55.570238</Coordinate><Coordinate>95.571596</Coordinate><Coordinate>0.000000</Coordinate>
        </CartesianPoint>
     </PolyLoop>
   </PlanarGeometry>
 </Opening>              
</Surface>

我从这个 - Xpath to select only nodes where child elements exist? SO线程中得到的参考很少,并且从下面的示例中得到的帮助很少。

book[author/degree]
All <book> elements that contain <author> children that in turn contain at least one <degree> child.

我如何使用xPath或其他方式实现这一目标?

4 个答案:

答案 0 :(得分:2)

  

我正在尝试获取其<Opening>的{​​{1}}具有笛卡尔点数&gt;的所有<PlanarGeometry>代码4。

假设<Polyloop>元素是您当前的上下文节点,那么:

Surface

gb:Opening[gb:PlanarGeometry/gb:Polyloop[count(gb:CartesianPoint) > 4]] 前缀需要映射到gb命名空间URI。这将选择包含至少一个http://www.gbxml.org/schema且包含4个Opening个孩子的所有Polyloop元素。

答案 1 :(得分:1)

以下XPath应该有效:

//g:Opening[4<count(./g:PlanarGeometry/g:PolyLoop/g:CartesianPoint)]

请注意,它使用名称空间前缀,因为Surface标记具有名称空间。我不太了解C#,但您可能必须先注册前缀才能使用它。

答案 2 :(得分:1)

请试一试:

/gb:Surface/gb:Opening[count(gb:PlanarGeometry/gb:PolyLoop/gb:CartesianPoint) > 4]

如此处所示,由于您的XML使用命名空间,因此您需要将该命名空间声明为XPath引擎,然后通过前缀引用它。它不一定是gb,但它必须是某种东西。

答案 3 :(得分:1)

如果您想使用LINQ to XML,那么它就是:

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

namespace ConsoleApplication1
{
    public class Program
    {
        public static void Main()
        {
            XElement sample = XElement.Load("c:\\sample.xml");
            IEnumerable<XElement> open_elements = sample.Descendants().Where(c => c.Name.LocalName == "Opening").Where(c => c.Descendants().Where(d => d.Name.LocalName == "CartesianPoint").Count() > 4);
            foreach (XElement ele in open_elements){
                Console.Write(ele.Attribute("id"));
            }
            Console.ReadKey();
        }
    }
}

希望这有帮助。