Java怎么用XPath解析带默认命名空间的XML

admin 百科 12
必须显式声明并绑定命名空间前缀,因XPath 1.0不支持默认命名空间;直接写/*/book会失败,因无前缀元素实际属于{http://example.com/ns}命名空间,而表达式book被解析为无命名空间节点,导致匹配失败。

Java怎么用XPath解析带默认命名空间的XML-第1张图片-佛山资讯网

Java用XPath解析带默认命名空间的XML,关键在于**必须显式声明并绑定命名空间前缀**,因为XPath 1.0(JDK内置支持的版本)不支持无前缀的默认命名空间匹配。

为什么直接写 /*//book 会失败?

XML中像 <root xmlns="http://example.com/ns"></root> 这样的默认命名空间,会让所有无前缀的元素都属于该命名空间。而XPath表达式中的 book 被视为“无命名空间的book”,和实际的 {http://example.com/ns}book 不匹配,结果就是查不到节点。

正确做法:注册命名空间前缀并使用它

需要两步:一是创建 NamespaceContext 实现类,把前缀(如 ns)映射到URI;二是在XPath编译前设置它。

  • 定义一个简单的命名空间上下文:

NamespaceContext nsContext = new NamespaceContext() {
    public String getNamespaceURI(String prefix) {
        if ("ns".equals(prefix)) return "http://example.com/ns";
        return null;
    }
    public String getPrefix(String namespaceURI) { return null; }
    public Iterator<String> getPrefixes(String namespaceURI) { return Collections.emptyIterator(); }
};

登录后复制

  • 设置到XPath对象上,并用带前缀的表达式查询:

XPath xpath = XPathFactory.newInstance().newXPath();
xpath.setNamespaceContext(nsContext);
Node book = (Node) xpath.evaluate("/ns:root/ns:book", doc, XPathConstants.NODE);

登录后复制

更简洁的方式:用 XPathConstants.NODESET 批量取节点

比如提取所有 title 元素:

标签: java node apache win 为什么

发布评论 0条评论)

还木有评论哦,快来抢沙发吧~