无法评估XPath中的表达式

Muh*_*aat 5 java xml xpath exception transformer-model

我使用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;
}
}
Run Code Online (Sandbox Code Playgroud)

例外情况:

javax.xml.transform.TransformerException: Unable to evaluate expression using this context
Run Code Online (Sandbox Code Playgroud)

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;
    }
}
Run Code Online (Sandbox Code Playgroud)

yan*_*kee 9

null尝试评估XPath表达式时,文档也可能导致"无法使用此上下文评估表达式"的异常.(我有同样的错误,我花了一段时间才弄清楚我没有正确初始化我的文档).

在您的代码中

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

// happily continue
Run Code Online (Sandbox Code Playgroud)

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

作为你的问题的读者,我甚至不知道抛出异常的位置.您应该提供此信息.请注意,您还可以使用someException.printStackTrace();堆栈跟踪指向正确的行.


Use*_*er0 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;
    }

}
Run Code Online (Sandbox Code Playgroud)

然后像这样启动 XPath 实例:

    getXPath().setNamespaceContext(new NamespaceResolver(doc));
Run Code Online (Sandbox Code Playgroud)