我想解析一个xml文件,我正在使用Java和xpath评估。
我想用xpath表达式输出当前的节点兄弟,而不是 使用getNextSibling()函数,可能吗?
e.g。如果我们读取name元素,我想添加xpath expression =“./ 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());
}
}
}
}
先谢谢
答案 0 :(得分:2)
首先,如果您使用正确形成的xml运行代码,应该关闭名称地址和专业标记,或者在尝试解析它时会出错,这将是一件好事。其次,如果你想选择节点的文本子节点,你需要确保那里有实际的东西,所以你的xml应该是这样的:
<root>
<name>hi</name>
<address>hi</address>
<profession>hi</profession>
</root>
现在,您选择name元素的文本很好,所以从IS that possible here ??
评论开始,您需要做出一些更改。
如果要评估相对XPath,则不希望将文档对象传递给XPath的evaluate方法。正在评估XPath的当前位置由您提供给它的项目确定,并且始终在根目录中评估文档对象。如果要评估相对于特定节点,而不是给它提供文档,请为其提供该节点。
因此,对于您的evaluate方法调用,您会有类似的内容:
Object result1 = expr.evaluate(nodes.item(i), XPathConstants.NODESET);
接下来,您应该确保您的XPath实际上是正确的。我们当前选择的节点是name的文本节点。这意味着我们需要首先转到名称节点而不是文本节点。 XPath语法中的.
表达式选择当前节点,因此您所做的就是选择相同的节点。您需要..
表达式来选择父节点。
因此,对于我们当前的..
XPath,我们选择名称节点。我们要做的是选择名称节点的sibilings的地址节点。我们有两种方法可以做到这一点,我们可以选择名称节点的父节点,根节点和选择其子节点的地址节点,或者我们可以使用XPath轴来选择兄弟节点(有关轴的信息可以是在这里找到http://www.w3schools.com/xpath/xpath_axes.asp)
如果我们要通过根节点,我们需要选择当前节点的父节点的父节点,以便../..
为我们提供根节点,然后是地址子节点:../../address/text()
,这将给我们所有地址兄弟姐妹。
或者,使用轴,我们可以..
选择名称节点,然后选择../following-sibling::address
(注意:这仅适用于地址节点之后名称 node)然后用../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);