如何使用Xpath检索XML树节点后的节点?

JAN*_*JAN 3 java xml xpath xml-parsing

首先,我必须说我发现它Xpath是一个非常好的解析器,我认为将它与其他解析器进行比较时非常强大.

给出以下代码:

  DocumentBuilderFactory domFactory = 
  DocumentBuilderFactory.newInstance();
  domFactory.setNamespaceAware(true); 
  DocumentBuilder builder = domFactory.newDocumentBuilder();
  Document doc = builder.parse("input.xml");
  XPath xpath = XPathFactory.newInstance().newXPath();
Run Code Online (Sandbox Code Playgroud)

如果我想找到first第1轮和第1轮的节点,请点击此处:

<Game>
    <Round>
        <roundNumber>1</roundNumber>
        <Door>
            <doorName>abd11</doorName>
            <Value>
                <xVal1>0</xVal1>
                <xVal2>25</xVal2>
                <pVal>0.31</pVal>
            </Value>
            <Value>
                <xVal1>25</xVal1>
                <xVal2>50</xVal2>
                <pVal>0.04</pVal>
            </Value>
            <Value>
                <xVal1>50</xVal1>
                <xVal2>75</xVal2>
                <pVal>0.19</pVal>
            </Value>
            <Value>
                <xVal1>75</xVal1>
                <xVal2>100</xVal2>
                <pVal>0.46</pVal>
            </Value>
        </Door>
        <Door>
            <doorName>vvv1133</doorName>
            <Value>
                <xVal1>60</xVal1>
                <xVal2>62</xVal2>
                <pVal>1.0</pVal>
            </Value>
        </Door>
    </Round>
    <Round>
        <roundNumber>2</roundNumber>
        <Door>
            <doorName>eee</doorName>
            <Value>
                <xVal1>0</xVal1>
                <xVal2>-25</xVal2>
                <pVal>0.31</pVal>
            </Value>
            <Value>
                <xVal1>-25</xVal1>
                <xVal2>-50</xVal2>
                <pVal>0.04</pVal>
            </Value>
            <Value>
                <xVal1>-50</xVal1>
                <xVal2>-75</xVal2>
                <pVal>0.19</pVal>
            </Value>
            <Value>
                <xVal1>-75</xVal1>
                <xVal2>-100</xVal2>
                <pVal>0.46</pVal>
            </Value>
        </Door>
        <Door>
            <doorName>cc</doorName>
            <Value>
                <xVal1>-60</xVal1>
                <xVal2>-62</xVal2>
                <pVal>0.3</pVal>
            </Value>
            <Value>
                <xVal1>-70</xVal1>
                <xVal2>-78</xVal2>
                <pVal>0.7</pVal>
            </Value>
        </Door>
    </Round>
</Game>
Run Code Online (Sandbox Code Playgroud)

我会这样做的:

 XPathExpression expr = xpath.compile("//Round[1]/Door[1]/Value[1]/*/text()");      
  Object result = expr.evaluate(doc, XPathConstants.NODESET);
  NodeList nodes = (NodeList) result;
Run Code Online (Sandbox Code Playgroud)

如果我想要second第1轮和第1轮的节点,那么:

XPathExpression expr = xpath.compile("//Round[1]/Door[1]/Value[2]/*/text()");  
Run Code Online (Sandbox Code Playgroud)

但是我如何使用循环来做这个,因为我不知道Value-nodes我有多少,这意味着我怎么能用循环来做这个,每次迭代我检索3(我的意思是xVal1,xVal2pVal值)更多值的值节点!?

要求这个的原因是:

  1. 我不知道Round我有多少

  2. 我不知道Value我有多少

  3. 我不想每次都申报新的 XPathExpression

谢谢 .

eri*_*xiv 7

选项1 - 迭代文档中的所有Value元素.只需要进行一次评估,但很难知道该值属于哪个Round或Door元素.

NodeList result = (NodeList) xpath.evaluate("//Round/Door/Value/*/text()", doc, XPathConstants.NODESET);
Run Code Online (Sandbox Code Playgroud)

选项2 - 分别迭代每个Round,Door和Value元素.需要更多评估,但上下文很容易知道.如果需要索引,则很容易在循环中添加计数器.

// Get all rounds and iterate over them
NodeList rounds = (NodeList) xpath.evaluate("//Round", doc, XPathConstants.NODESET);
for (Node round : rounds) {
  // Get all doors and iterate over them
  NodeList doors = (NodeList) xpath.evaluate("Door", round, XPathConstants.NODESET);
  for (Node door : doors) {
    // Get all values and iterate over them
    NodeList values = (NodeList) xpath.evaluate("Value/*/text()", door, XPathConstants.NODESET);
    for (Node value : values) {
      // Do something
    }
  }
}
Run Code Online (Sandbox Code Playgroud)

选项3 - 根据您的要求做上述的一些组合

请注意,我已删除表达式编译步骤以缩短示例.应该重新添加它以提高性能.

  • Nodelist不能像那样迭代. (3认同)
  • 您正在使用哪个`NodeList`类来实现`Iterable`? (2认同)
  • @majorbanzai您是否在考虑增强的for循环语法?我怀疑您是对的,因为“ NodeList”不是“ Iterable”,因此代码示例将不起作用。确实使示例更具可读性... (2认同)