我为简单的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);

10-08 14:58