8

我想对此文件进行 XPath 查询(显示摘录):

<?xml version="1.0" encoding="UTF-8"?>
<!-- MetaDataAPI generated on: Friday, May 25, 2007 3:26:31 PM CEST -->
<ModelClass xmlns="http://xml.sap.com/2002/10/metamodel/webdynpro" xmlns:IDX="urn:sap.com:WebDynpro.ModelClass:2.0">
    <ModelClass.Parent>
        <Core.Reference package="com.test.mypackage" name="ModelName" type="Model"/>

这是我正在使用的代码片段:

DocumentBuilderFactory domFactory = DocumentBuilderFactory.newInstance();
DocumentBuilder builder = domFactory.newDocumentBuilder();
Document document = builder.parse(new File(testFile));
XPathFactory factory = XPathFactory.newInstance();
XPath xpath = factory.newXPath();
xpath.setNamespaceContext( new NamespaceContext() {
    public String getNamespaceURI(String prefix) {
...

String result = xpath.evaluate(xpathQueryString, document);
System.out.println(result);

我面临的问题是,在 XPath 查询中引用默认命名空间时,不会调用 getNamespaceURI 方法来解决它。例如,此查询不提取任何内容:

//xmlns:ModelClass.Parent/xmlns:Core.Reference[@type=\"Model\"]/@package

现在我尝试通过替换xmlns为假前缀来“欺骗”解析器d,然后getNamespaceURI相应地编写方法(以便在遇到http://xml.sap.com/2002/10/metamodel/webdynpro时返回d)。在这种情况下,getNamespaceURI会调用 ,但 XPath 表达式求值的结果始终是空字符串。

如果我从文件和 XPath 查询表达式中去除名称空间,我可以得到我想要的字符串 (com.test.mypackage)。

有没有办法让默认命名空间正常工作?

4

2 回答 2

13

XPath 1.0 规范要求“没有前缀意味着没有命名空间”。因此,为 XPath 1.0 设计的 JAXP 阻止您将“空前缀”绑定到某个非空名称空间是完全正确的。

XPath 2.0 允许您在 XPath 表达式中为非限定名称声明一个默认名称空间,但要利用它,您需要一个利用此功能的 API(例如 Saxon 的 s9api)。

于 2012-05-23T16:19:10.907 回答
11

在您的Namespace上下文中,将您选择的前缀(例如df)绑定到文档中的命名空间 URI

xpath.setNamespaceContext( new NamespaceContext() {
    public String getNamespaceURI(String prefix) {
      switch (prefix) {
        case "df": return "http://xml.sap.com/2002/10/metamodel/webdynpro";
        ...
       }
    });

然后在路径表达式中使用该前缀来限定元素名称,例如/df:ModelClass/df:ModelClass.Parent/df:Core.Reference[@type = 'Model']/@package.

于 2012-05-23T13:26:05.067 回答