5

我正在使用 XPath 解析由 URL 返回的 XML 文档,当我使用给定的输入运行我的代码时,它可以工作,但是当它作为用户输入提供输入时,它会引发异常。 编码:

    class{
        private String generalQuery =  "//@*";
    method(){
        System.out.println("Enter URL");
        url = scan.nextLine();
        URL oracle = new URL(url);
        InputStream is = oracle.openStream();

        org.w3c.dom.Document doc = null;
        DocumentBuilderFactory domFactory;
        DocumentBuilder builder;

        try {
            domFactory = DocumentBuilderFactory.newInstance();
            domFactory.setNamespaceAware(true);
            builder = domFactory.newDocumentBuilder();
            doc = builder.parse(is);
        } catch (Exception ex) {
            System.err.println("unable to load XML: " + ex);
        }

    Map <String, String> params = new HashMap<String, String> ();

            XPathFactory factory = XPathFactory.newInstance();
            XPath xpath = factory.newXPath();
            xpath.setNamespaceContext(new NameSpaces(doc));
            XPathExpression expr = xpath.compile(generalQuery);
            Object result = expr.evaluate(doc, XPathConstants.NODESET); // exception thrown here

            NodeList nl = (NodeList) result;

            for (int i = 0 ; i < nl.getLength() ; i++){
                Node n = (Node)nl.item(i);
                params.put(n.getNodeName(), n.getNodeValue());
            }

            return params;
}
}

例外:

javax.xml.transform.TransformerException: Unable to evaluate expression using this context

类 NameSpaces :

import java.util.Iterator;
import javax.xml.XMLConstants;
import javax.xml.namespace.NamespaceContext;
import org.w3c.dom.Document;

public class NameSpaces implements NamespaceContext {
    private Document sourceDocument;

    public NameSpaces(Document document) {
        sourceDocument = document;
    }

    @Override
    public String getNamespaceURI(String prefix) {
        if (prefix.equals(XMLConstants.DEFAULT_NS_PREFIX)) {
            return sourceDocument.lookupNamespaceURI(null);
        } else {
            return sourceDocument.lookupNamespaceURI(prefix);
        }
    }

    @Override
    public String getPrefix(String namespaceURI) {
        return sourceDocument.lookupPrefix(namespaceURI);
    }

    @Override
    public Iterator<String> getPrefixes(String namespaceURI) {
        return null;
    }
}
4

3 回答 3

10

null尝试评估 XPath 表达式时,文档也可能会导致“无法使用此上下文评估表达式”异常。(我遇到了同样的错误,我花了一段时间才发现我没有正确初始化我的文档)。

在您的代码中,您有

try {
  // load document
}
catch (Exception ex) {
  System.err.println("unable to load XML: " + ex);
}

// happily continue

这是自找麻烦。如果在初始化期间发生异常,您应该立即停止并且不应该继续。如果您完全不知道如何处理错误,请使用catch(Exception e) { throw new Error(e); }. 这将导致异常冒泡,并希望由打印堆栈跟踪并退出的默认异常处理程序处理。

作为您问题的读者,我什至不知道异常是在哪里引发的。您应该提供这些信息。请注意,您还可以使用someException.printStackTrace();获取指向正确行的堆栈跟踪。

于 2014-02-13T12:12:05.080 回答
0

您似乎缺少的是NameSpaceContext您可以自己实现的。

另请参阅此线程:NamespaceContext and using namespaces with XPath

例子:

class NamespaceResolver implements NamespaceContext {

    private final Document document;

    public NamespaceResolver(Document document) {
        this.document = document;
    }

    public String getNamespaceURI(String prefix) {
        if (prefix.equals(XMLConstants.DEFAULT_NS_PREFIX)) {
            return document.lookupNamespaceURI(null);
        } else {
            return document.lookupNamespaceURI(prefix);
        }
    }

    public String getPrefix(String namespaceURI) {
        return document.lookupPrefix(namespaceURI);
    }

    @SuppressWarnings("rawtypes")
    public Iterator getPrefixes(String namespaceURI) {
        // not implemented
        return null;
    }

}

然后像这样启动 XPath 实例:

    getXPath().setNamespaceContext(new NamespaceResolver(doc));
于 2013-05-20T14:51:06.683 回答
0

在我的情况下,这不是由于null文档,而是由于没有根元素的空文档。附加后者解决了这个问题。

于 2018-06-15T10:56:34.207 回答