Xalan问题与简单的XSL转换

时间:2011-06-08 06:41:36

标签: java xslt xalan

我对一个简单的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源作为String(通过StringReader和StreamSource)提供,它也适用于Xalan。我开始怀疑问题出在DOMSource中......

2 个答案:

答案 0 :(得分:4)

尝试为DocumentBuilder启用名称空间,因为我看到你在xslt匹配中使用名称空间。还使用document而不是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);

答案 1 :(得分:2)

没有看到XSLT有什么问题,但如果没有使用的XML源,很难说。

但如果我不得不猜测,我认为问题出在名称空间中。源文档是否也定义并使用该xhtml命名空间......?它是用名称空间感知的XML解析器解析的吗?