我为简单的XSL转换而生气。经过大量测试,我发现它取决于Xalan(因为使用Saxon可以工作)。我已经确定Java必须使用Xalan处理器。
唯一匹配的规则是“身份规则”,而其他“匹配”没有完成。如果我使用撒克逊,一切正常!
您可以在此XSL transf中看到错误吗?
<?xml version="1.0" encoding="UTF-8"?>
<xsl:stylesheet version="1.0" xmlns:xhtml="http://www.w3.org/1999/xhtml"
xmlns:xsl="http://www.w3.org/1999/XSL/Transform"
exclude-result-prefixes="xhtml xsl">
<xsl:output method="xml" omit-xml-declaration="no"
media-type="text/xml" indent="yes" encoding="UTF-8"
doctype-public="-//W3C//DTD XHTML 1.1//EN" doctype-system="http://www.w3.org/TR/xhtml11/DTD/xhtml11.dtd" />
<xsl:template match="/">
<xsl:message>root</xsl:message>
<xsl:apply-templates select="xhtml:html" />
</xsl:template>
<xsl:template match="xhtml:html">
<xsl:element name="html">
<xsl:attribute name="xml:lang">it</xsl:attribute>
<xsl:apply-templates select="xhtml:head|xhtml:body" />
</xsl:element>
<xsl:message>xhtml:html</xsl:message>
</xsl:template>
<xsl:template match="xhtml:head">
<xsl:element name="head">
<xsl:apply-templates select="xhtml:meta" />
<title>
<xsl:value-of select="xhtml:title" />
</title>
<link href="%stile.css%" rel="stylesheet" type="text/css" />
<xsl:apply-templates select="xhtml:script" />
</xsl:element>
<xsl:message>xhtml:head</xsl:message>
</xsl:template>
<xsl:template match="xhtml:body">
<xsl:element name="body">
<xsl:apply-templates select="descendant::xhtml:div[@class='sxSmall']" />
</xsl:element>
<xsl:message>xhtml:body</xsl:message>
</xsl:template>
<xsl:template match="xhtml:script">
<xsl:message>xhtml:script</xsl:message>
</xsl:template>
<xsl:template match="xhtml:meta">
<xsl:message>xhtml:meta</xsl:message>
</xsl:template>
<!-- identity -->
<xsl:template match="@*|node()">
<xsl:copy>
<xsl:apply-templates select="@*|node()" />
</xsl:copy>
<xsl:message>identita</xsl:message>
</xsl:template>
<xsl:template match="xhtml:div[@class='sxSmall']">
<xsl:element name="div">
<xsl:attribute name="class"><xsl:value-of select="@class" /></xsl:attribute>
<xsl:apply-templates />
</xsl:element>
<xsl:message>xhtml:div</xsl:message>
</xsl:template>
</xsl:stylesheet>
--------------更新----------------
问题与Xalan以及将XML源放入其中的方式有关。 Java代码如下:
StringReader srXslContent = new StringReader(xslContent);
TransformerFactory tFactory = TransformerFactory.newInstance("net.sf.saxon.TransformerFactoryImpl",null);
Transformer transformer = null;
try
{
transformer = tFactory.newTransformer(new StreamSource(srXslContent));
Context.getInstance().getLogger().debug("Transformer created!");
} catch (TransformerConfigurationException e)
{
Context.getInstance().getLogger().error(ExceptionUtils.getStackTrace(e));
}
StringWriter sw = new StringWriter();
Document doc = XmlUtils.parseXmlFile(xmlContent);
DOMSource domSource = new DOMSource(doc.getDocumentElement());
try
{ // Could be this?
transformer.transform(domSource, new StreamResult(sw));
// with streamsource Xalan works fine!!!
//transformer.transform(new StreamSource(new StringReader(xmlContent)), new StreamResult(sw));
Context.getInstance().getLogger().debug("Transformation made!");
} catch (TransformerException e)
{
Context.getInstance().getLogger().error(ExceptionUtils.getStackTrace(e));
}
parseXml方法在这里:
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(false);
dbf.setValidating(false);
// Sax
dbf.setFeature("http://xml.org/sax/features/validation", false);
// Xerces: to disable Internet searching...
dbf.setFeature("http://apache.org/xml/features/nonvalidating/load-dtd-grammar", false);
dbf.setFeature("http://apache.org/xml/features/nonvalidating/load-external-dtd", false);
DocumentBuilder db = dbf.newDocumentBuilder();
InputSource is = new InputSource(new StringReader(contentToParse));
Document document =db.parse(is);
return document;
重要的是要注意,如果我将XML源作为字符串(通过StringReader和StreamSource)提供给Xalan,它也可以正常工作。我开始怀疑问题出在DOMSource中...
最佳答案
尝试为DocumentBuilder启用名称空间,因为我看到您在xslt匹配项中使用名称空间。还要使用文档而不是document.getDocumentElement()方法构造新的DOMSource。
DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance();
dbf.setNamespaceAware(true);
dbf.setValidating(false);
DocumentBuilder db = dbf.newDocumentBuilder();
Document d = db.parse(...);
DOMSource ds = new DOMSource(d);