无法评估XPath中的表达式

时间:2013-05-20 11:19:02

标签: java xml exception xpath transformer

我;使用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;
    }
}

3 个答案:

答案 0 :(得分:9)

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

在您的代码中

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();来获取指向正确行的堆栈跟踪。

答案 1 :(得分: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));

答案 2 :(得分:0)

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