创建
XSLT 的第一个版本有很大的局限性,只能有一个输入和一个输出(虽然可以有多个模板文件)。该标准的第 2 版仍然要求一个输入,但是输出系统更加灵活了。现在可以使用 xsl:result-document
指令产生多个输出文件。这一新的标签有两个重要属性,如表 1 所示。
属性 | 说明 |
href | 输出文件的文件名或者完全限定的 URL |
format | 使用的格式名,如对应 xsl:output 指令中所定义的 |
为了试验该指令,我设计了一个输入 XML 文件,包括一组测试结果(参见清单 1)。
<?xml version="1.0" encoding="UTF-8"?><tests> <testrun run="test1"> <test name="foo" pass="true" /> <test name="bar" pass="true" /> <test name="baz" pass="true" /> </testrun> <testrun run="test2"> <test name="foo" pass="true" /> <test name="bar" pass="false" /> <test name="baz" pass="false" /> </testrun> <testrun run="test3"> <test name="foo" pass="false" /> <test name="bar" pass="true" /> <test name="baz" pass="false" /> </testrun></tests>
|
这段 XML 非常简单。每个测试运行中都有一组带有 pass 标记的指定测试,用以说明测试是否成功。
首先为每个测试结果创建一个文件,XSL 模板如清单 2 所示。
<?xml version="1.0" encoding="UTF-8"?><xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/XSL/Transform" version="2.0"><xsl:output method="text"/><xsl:output method="html" indent="yes" name="html"/><xsl:template match="/"><xsl:for-each select="//testrun"><xsl:variable name="filename" select="concat('output1/',run,'.html')" /><xsl:value-of select="$filename" /> <!-- Creating --><xsl:result-document href="{$filename}" format="html"> <html><body> <xsl:value-of select="run"/> </body></html></xsl:result-document></xsl:for-each></xsl:template></xsl:stylesheet>
|
有几个地方值得注意,从文件头开始说起。stylesheet 标签的 version
属性被设为 2.0
,这样就能够使用 xsl:result-document
标签了。接下来,样式表本身设置 text
作为输出类型。这意味着如果希望让这个文件具有 HTML 格式,那么还需要定义类型 html
的第二个指定格式。我在 xsl:result-document
标签中使用这种格式。
然后使用 xsl:for-each
循环遍历 testrun
标签。在每个标签中使用 variable
标签创建一个新的 $filename
变量,将输出目录名(output1
)、 运行名和扩展名 .html 合成一个路径。
然后我使用 value-of
标签和 $filename
变量告诉用户正在创建的文件。再用 xsl:result-document
标签打开新文档输出 HTML。清单 3 显示了运行上述数据文件时的结果。
Creating output1/test1.htmlCreating output1/test2.htmlCreating output1/test3.html
|
xsl:result-document
标签的内容按照与模板其他部分相同的方式计算似乎有点不可思议,但确实如此。模板上下文中的所有变量都可以使用。
作为一个例子,我修改了 xsl:result-document
标签中的代码,以便提供关于测试结果的更多信息(参见清单 4)。
<xsl:result-document href="{$filename}" format="html"> <html><head> <title>Test results - <xsl:value-of select="run"/></title> </head><body> <xsl:value-of select="run"/> <!-- Run --> <table> <tr><td>Test</td><td>Pass</td></tr> <xsl:for-each select="test"> <tr><td> <xsl:value-of select="name" /> </td><td> <xsl:value-of select="pass" /> </td></tr> </xsl:for-each> </table> </body></html></xsl:result-document>
|
输出结果如清单 5 所示。
<html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Test results - test1</title> </head> <body> <!-- Run: test1--> <table> <tr> <td>Test</td> <td>Pass</td> </tr> <tr> <td>foo</td> <td>true</td> </tr> <tr> <td>bar</td> <td>true</td> </tr> <tr> <td>baz</td> <td>true</td> </tr> </table> </body></html>
|
最后还需要增加一个索引文件指向输出的所有测试结果。为此,我使用了另一个 xsl:result-document
标签,并将输出硬编码到索引文件中(参见清单 6)。
<!-- Creating the index --><xsl:result-document href="output3/index.html" format="html"> <html><head><title>Test Index</title></head> <body> <xsl:for-each select="//testrun"> <a href="{run}.html"><xsl:value-of select="run" /> </a><br/> </xsl:for-each> </body> </html></xsl:result-document>
|
这一段放在为每个测试案例建立 HTML 文件的 xsl:for-each
循环之后。清单 7 显示了得到的索引文件。
<html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Test Index</title> </head> <body><a href="test1.html">test1</a> <a href="test2.html">test2</a> <a href="test3.html">test3</a> </body></html>
|
使用 xsl:result-document
指令可以将一个 XSL 模板从单个数据源输出到多个文件中。这种功能在 XSLT 1.x 属于非标准扩展,为 XSLT 模板创造者打开了新的机会之窗。