【问题标题】:Parse XML file using XPATH expressions and Java使用 XPATH 表达式和 Java 解析 XML 文件
【发布时间】:2013-09-19 12:58:15
【问题描述】:

我想解析一个 xml 文件,我正在使用 Java 和 xpath 评估。

我想用 xpath 表达式输出当前的节点兄弟,而不是 使用 getNextSibling() 函数,可以吗?

例如如果我们读取名称元素,我想添加 xpath 表达式 =“./address” 为了在不使用 getNextSibling()/ 的情况下输出“name”的兄弟姐妹。

xml文件如下:

  <root>            
        <name>
        <address>
        <profession>
  </root>

我的代码如下:

  package dom_stack4;

  import org.w3c.dom.*;
  import javax.xml.xpath.*;
  import javax.xml.parsers.*;
  import java.io.IOException;
  import org.xml.sax.SAXException;


  public class Dom_stack4 {

public static void main(String[] args) throws ParserConfigurationException,       SAXException, 
    IOException, XPathExpressionException {
    // TODO code application logic here


    DocumentBuilderFactory domFactory = 
    DocumentBuilderFactory.newInstance();
    domFactory.setNamespaceAware(true); 
    DocumentBuilder builder = domFactory.newDocumentBuilder();
    Document doc = builder.parse("root.xml");
    XPath xpath = XPathFactory.newInstance().newXPath();
     // XPath Query for showing all nodes value
    XPathExpression expr = xpath.compile("/root/name/text()");

    Object result = expr.evaluate(doc, XPathConstants.NODESET);
    NodeList nodes = (NodeList) result;
    for (int i = 0; i < nodes.getLength(); i++) {
          System.out.println(" Name is : " +  nodes.item(i).getNodeValue()); 

          /* IS that possible here ?? */
         /* ./address/text() => outputs the current's node sibling */
          expr = xpath.compile("./address/text()");
          Object result1 = expr.evaluate(doc, XPathConstants.NODESET);
          NodeList nodes1 = (NodeList) result1;
           for (int j = 0; j < nodes1.getLength(); j++) {
               System.out.println(" ---------------- " + nodes1.item(j).getNodeValue()); 
              }

    }
    }
    }

谢谢,提前

【问题讨论】:

    标签: java xml parsing dom


    【解决方案1】:

    首先,如果您使用格式正确的 xml 运行代码会很好,名称地址和职业标签应该关闭,否则当您尝试解析它时会出错。其次,如果你想选择节点的文本子节点,你需要确保那里确实有一些东西,所以你的 xml 应该看起来像这样:

    <root>
        <name>hi</name>
        <address>hi</address>
        <profession>hi</profession>
    </root>
    

    现在,您可以选择名称元素的文本,所以从您的 IS that possible here ?? 评论开始,您需要进行一些更改。

    如果你想计算一个相对的 XPath,你不想将你的文档对象传递给 XPath 的评估方法。评估 XPath 的当前位置由您提供给它的项目确定,并且文档对象始终在根处评估。如果您想相对于特定节点进行评估,而不是给它文档,给它那个节点。

    所以你的评估方法调用会有这样的东西:

    Object result1 = expr.evaluate(nodes.item(i), XPathConstants.NODESET);
    

    接下来,您应该确保您的 XPath 实际上是正确的。我们当前选择的节点是名称的文本节点。这意味着我们需要首先转到名称节点而不是文本节点。 XPath 语法中的. 表达式选择当前节点,因此您所做的只是选择同一个节点。您想要选择父节点的.. 表达式。

    因此,使用我们当前的 XPath ..,我们正在选择名称节点。我们要做的是选择作为名称节点的兄弟的地址节点。有两种方法可以做到这一点,我们可以选择名称节点的父节点,根节点,并选择作为其子节点的地址节点,或者我们可以使用 XPath 轴来选择兄弟节点(关于轴的信息可以是在这里找到http://www.w3schools.com/xpath/xpath_axes.asp)

    如果我们要通过根节点,我们需要选择当前节点的父节点的父节点,所以../.. 给出了根节点,然后是地址子节点:../../address/text(),这将给出我们都称呼兄弟姐妹。

    或者,使用轴,我们可以使用.. 来选择名称节点,然后选择../following-sibling::address(注意:这仅适用于地址节点在名称节点之后),然后选择带有../following-sibling:address/text()的地址节点的文本。

    这给了我们这两行

    expr = xpath.compile("../../address/text()"); 
    Object result1 = expr.evaluate(nodes.item(i), XPathConstants.NODESET);
    

    expr = xpath.compile("../following-sibling::address/text()");
    Object result1 = expr.evaluate(nodes.item(i), XPathConstants.NODESET);
    

    【讨论】:

      猜你喜欢
      • 1970-01-01
      • 2013-02-20
      • 2013-06-13
      • 1970-01-01
      • 1970-01-01
      • 2010-10-05
      • 1970-01-01
      • 2012-06-02
      • 1970-01-01
      相关资源
      最近更新 更多