我将使用Java程序应用我的XSLT时看到的行为不一致与使用XML Spy进行测试相比。具体来说,使用Java时,xsl:for-each
指令似乎不会遍历整个列表。使用XML Spy进行测试时不是这种情况。
我的XML文档:
<metadata xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:fn="http://www.w3.org/2005/xpath-functions">
<maps>
<geographicCoverages>
<coverage key="ON">Ontario</coverage>
<coverage key="MB">Manitoba</coverage>
<coverage key="SK">Saskatchewan</coverage>
<coverage key="AB">Alberta</coverage>
<coverage key="BC">British Columbia</coverage>
<coverage key="YT">Yukon</coverage>
<coverage key="NU">Nunavut</coverage>
</geographicCoverages>
</maps>
<entry>
<string>GEO_COVERAGE</string>
<list>
<string>BC</string>
<string>ON</string>
</list>
</entry>
...
</metadata>
我的XSLT:
<xsl:stylesheet version="2.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:xs="http://www.w3.org/2001/XMLSchema" xmlns:fn="http://www.w3.org/2005/xpath-functions">
<xsl:output method="text" version="1.0" indent="yes"/>
<xsl:key name="geoCoverageKey" match="metadata/maps/geographicCoverages/coverage" use="@key" />
<xsl:template match="/metadata">
<xsl:for-each select="entry[string='GEO_COVERAGE']/list">
<xsl:value-of select="key('geoCoverageKey', string)" separator=", " />
</xsl:for-each>
</xsl:template>
</xsl:stylesheet>
当我用XMLSpy测试上面的XSLT时,我得到了我期望的输出:
Ontario, British Columbia
我的Java测试计划:
public static void main(String[] args) throws TransformerException {
String xsltPath = "test_migration.xslt";
String xmlPath = "test-migration.xml";
TransformerFactory tFactory = TransformerFactory.newInstance();
Transformer transformer = tFactory.newTransformer(new StreamSource(xsltPath));
StreamResult result = new StreamResult( new StringWriter() );
transformer.transform(new StreamSource(xmlPath), result );
System.out.println( result.getWriter().toString() );
}
但是,当我运行上面的Java测试程序时,我得到的输出只包含一个值;安大略省。我不知道为什么我会得到这种差异。
答案 0 :(得分:0)
JDK中的默认处理器是Xalan。 Xalan是一个XSLT 1.0处理器。当给XSLT 1.0处理器一个指定version =“2.0”的样式表时,需要(在XSLT 1.0处理器的规则下)忽略XSLT 1.0中未定义的属性。 “分隔符”属性属于此类别。在XSLT 1.0中,xsl:value-of输出节点集中的第一个节点,忽略其余节点。
答案是安装Saxon,它提供Java中的XSLT 2.0处理。