1

我正在对输入 XML 进行 XSL 转换,然后我需要在结果文档上使用 XPath 提取一些值。但是在使用 XSL 结果节点时,XPath 表达式可能总是返回 null。但是如果我将 XSL 生成的文档存储在一个文件中,然后重新加载它。XPath 表达式返回相应的节点。

这是我的代码(为了方便起见,实用程序已被删除):

public class XmlTest {
@Test
public void testWithNativeJavaApi() throws Exception {
    InputStream instream = resolveClasspathFile("xslt/xslt-test-transform-2.xsl");
    StreamSource xsltSource = new StreamSource(instream);
    DOMSource domSource = loadXmlFromClasspathFile("xslt/xslt-test-input-2.xml");
    prettyPrint(domSource.getNode());

    Transformer transformer = TransformerFactory.newInstance().newTransformer(xsltSource);
    DOMResult domResult = new DOMResult();
    transformer.transform(domSource, domResult);
    Node node = domResult.getNode();

    // Store then reload the file
    // Uncommenting those 3 lines will make the test pass
    // File xslOutputfile = new File("target", "xsl-ouput.xml");
    // prettyPrint(node, new FileOutputStream(xslOutputfile));
    // node = loadXmlFromInputStream(new FileInputStream(xslOutputfile)).getNode();

    XPath xPathProcessor = XPathFactory.newInstance().newXPath();
    XPathExpression xpathExpression = xPathProcessor.compile("/Message/Out/Personne/CodeCivilite");
    System.out.println();
    Node resultNode = (Node) xpathExpression.evaluate(node, XPathConstants.NODE);

    if (resultNode != null) {
        System.out.println(resultNode.getNodeName() + "=" + resultNode.getTextContent());
    } else {
        System.out.println("Node is null");
    }

    assertNotNull("XPath expression returned null node", resultNode);
    assertEquals("CodeCivilite", resultNode.getNodeName());
    assertEquals("M.", resultNode.getTextContent());

}
}

只需注释或删除“// Store 然后重新加载文件”下面的 3 行,测试将不再通过。

我完全陷入困境,欢迎任何帮助。

4

3 回答 3

1

DocumentBuilderFactory 默认不使用命名空间,除非使用 dbf.setNamespaceAware(true)。另一方面,XSL 和 TransformerFactory 大量使用命名空间,因此转换的结果文档使用命名空间进行限定。

因此,在 XSL 转换旁边调用 XPathExpression 时应该使用名称空间。

XPath xPathProcessor = XPathFactory.newInstance().newXPath();
xPathProcessor.setNamespaceContext(new NamespaceContext() {
  @Override
  public String getNamespaceURI(String prefix) {
    if(prefix.equals("t")) {
      return "http://www.example.org/test";
    } else {
      return null;  
    }
  }
  @Override public String getPrefix(String namespaceURI) {
    return null; // not used
  }
  @Override
  public Iterator<?> getPrefixes(String namespaceURI) {
    return null; // not used
  }
});
XPathExpression xpathExpression = xPathProcessor.compile("/t:Message/t:Out/t:Personne/t:CodeCivilite");
System.out.println();
Node resultNode = (Node) xpathExpression.evaluate(node, XPathConstants.NODE);

引入一个临时文件,使用默认 DocumentBuilderFactory(不知道命名空间)重新加载,使非限定 XPATH 表达式(没有命名空间)工作。

希望这很清楚:)

于 2012-07-25T06:58:48.460 回答
0

我想知道解析文件是否会给你一个更短的树。尝试使用 xpath 表达式

//Message/Out/Personne/CodeCivilite

(前面的双斜线)没有文件 i/o 以查看是否有没有文件 i/o 的额外节点层。

于 2012-07-23T19:51:58.400 回答
0

节点 = loadXmlFromInputStream(new FileInputStream(xslOutputfile))。getDocumentElement();

是 getDocumentElement() 吗?

你能打印重新加载的节点,看看你是否有你所期望的吗?

于 2012-07-23T19:59:52.000 回答