XSL的作用:将XML转换成HTML
例子说明一切:下面是XML文档的一部分:

<?xml version="1.0" encoding="ISO8859-1" ?>
<CATALOG>
<CD>
<TITLE>Empire Burlesque</TITLE>
<ARTIST>Bob Dylan</ARTIST>
<COUNTRY>USA</COUNTRY>
<COMPANY>Columbia</COMPANY>
<PRICE>10.90</PRICE>
<YEAR>1985</YEAR>
</CD>
...

</CATALOG>

然后我们将下面的XSL文件作为HTML的模板将XML数据转换为HTML文件:

<?xml version='1.0'?>
<xsl:stylesheet xmlns:xsl="http://www.w3.org/TR/WD-xsl">
<xsl:template match="/">
<html>
<body>
<table border="2" bgcolor="yellow">
<tr>
<th>Title</th>
<th>Artist</th>
</tr>
<xsl:for-each select="CATALOG/CD">
<tr>
<td><xsl:value-of select="TITLE"/></td>
<td><xsl:value-of select="ARTIST"/></td>
</tr>
</xsl:for-each>
</table>
</body>
</html>
</xsl:template>
</xsl:stylesheet>

在上面的代码中, xsl:for-each元素的作用是定位XML文档中的哪些元素需要按以下模板显示。select属性用来定义源文件中的元素名。指定属性的这种语法又称为XML Pattern(模式),类似文件子目录的表示形式。xsl:value-of元素用来在当前层次中插入子元素的内容模板。
因为XSL样式表自身也是一个XML文档,因此,XSL文件的开头以一个XML声明开始。 xsl:stylesheet元素用来声明这是一个样式表文件。<xsl:template
match="/">语句表示XML的源文档在当前目录下。
如果为XML文档加上XSL样式表,看下面代码第2行,你的浏览器就可以精确的将XML 文档转换为HTML文件。

<?xml version="1.0" encoding="ISO8859-1" ?>
<?xml-stylesheet type="text/xsl" href="cd_catalog.xsl"?>
<CATALOG>
<CD>
<TITLE>Empire Burlesque</TITLE>
<ARTIST>Bob Dylan</ARTIST>
<COUNTRY>USA</COUNTRY>
<COMPANY>Columbia</COMPANY>
<PRICE>10.90</PRICE>
<YEAR>1985</YEAR>
</CD>

...

</CATALOG>

XSL 的索引

当XML文档被转换成HTML文件,索引应该同时建立。简单的办法就是给你的for-each元素增加一个order-by属性,就象这样:
<xsl:for-each select="CATALOG/CD" order-by="+ ARTIST">
order-by属性带有一个"+"或者"-" 的符号,用来定义索引的方式,是升序还是降序排列。符号后面的名字就是要索引的关键字。

XSL的过滤和查询

我们希望只显示满足一定的条件的XML数据应该怎么做呢?还是上面的例子代码,我们只需要在xsl:for-each元素的select属性中加入参数就可以,类似:
<xsl:for-each select="CATALOG/CD[ARTIST='Bob Dylan']">
参数的逻辑选择有:
= (等于)
=! (不等于)
&LT& 小于
&GT& 大于等于

 

XSL的控制语句

1.条件语句if...then
XSL同样还有条件语句(呵呵~~好厉害吧,象程序语言一样)。具体的语法是增加一个xsl:if元素,类似这样
<xsl:if match=".[ARTIST='Bob Dylan']">

... some output ...
</xsl:if>
2. XSL 的Choose
choose的用途是出现多个条件,给出不同显示结果。具体的语法是增加一组xsl:choose,xsl:when,xsl:otherwise元素:

<xsl:choose>
<xsl:when match=".[ARTIST='Bob Dylan']">
... some code ...
</xsl:when>
<xsl:otherwise>
... some code ....
</xsl:otherwise>
</xsl:choose>

 

XSLT

XSL和XSLT从狭义上理解是一样的,而按照W3C的标准,XSLT的说法更严格些。

其实很多和上面XSL是重复的,但是下面解释的更详细

XSLT的元素语法
1 xsl:template和xsl:apply-templates 

模板(template)是XSLT中最重要的概念之一。XSLT文件就是由一个一个的模板组成,任何一个XSLT文件至少包含一个模板。模板(template)由两部分组成:匹配模式(match pattern)和执行。简单的讲模式定义XML源文档中哪一个节点将被模板处理,执行则定义输出的是什么格式。两部分对应的语法为xsl:template和xsl:apply-templates。 

xsl:template的语法是: 

<xsl:template

match = pattern 

name = qname

priority = number

 mode = qname>
<!-- 执行内容 -->
</xsl:template> 

xsl:template的作用是定义一个新模板。属性中name,priority,和mode用来区别匹配同一节点的不同模板。match属性则控制模板的匹配模式(pattern),匹配模式是用来定位XML源文档中哪一个节点被模板处理。一个模板匹配一个节点。我们用一个例子来帮助理解:
假设我们要处理一个包含章节和段落文档。我们用para元素定义段落,用chapter元素定义章节。我们来看看match属性可能的值。下面的语句写法说明模板匹配所有的para元素 

<xsl:template match="para">
</xsl:template> 

下面的语句写法说明模板匹配所有的para元素和所有的chapter元素: 

<xsl:template match="(chapterpara)">
</xsl:template> 

下面的语句写法说明模板匹配所有的父节点为chapter元素的para元素: 

<xsl:template match="chapter//para">
</xsl:template> 

下面的语句写法说明模板匹配根节点: 

<xsl:template match="/">
</xsl:template> 

apply-templates语法:

<xsl:apply-templates
select = node set-expression
mode = qname>
</xsl:apply-templates> 

xsl:apply-templates用来执行那一个节点被模板具体处理。你可以将它理解为程序中调用子函数。select属性用来定义确切的节点名称。xsl:apply-templates总是包含在xsl:template元素中,象这样: 

<xsl:template match="/">
<xsl:apply-templates select="para"/>
</xsl:template> 

这段代码说明摸板匹配整个文档(根节点),具体执行时处理根节点下所有para元素。 

<xsl:template match="para">
<p><xsl:apply-templates/></p>
</xsl:template> 

而这一段代码则表示摸板匹配para节点,所有para下的子元素都将被处理。

xsl:value-of 

XSL:value-of用来将源文档中元素的文本值写到输出文档中。例如:
有一个个人资料的XML文档: 

<?xml version="1.0" encoding="iso-8859-1"?>
<PERSON>
<name>ajie</name>
<age>28</age>
</PERSON> 

我如果想在输出文档中显示上面这个XML源文档中的name元素的值,可以这样写XSLT代码: 

<xsl:template match="PERSON">
<xsl:value-of select="name"/>
</xsl:template>

执行后,你会看到"ajie"被单独显示出来。其中match="PERSON"定义摸板匹配PERSON节点,xsl:value-of 语法说明需要输出一个节点的值,而select="name"则定义需要被输出的元素为name。看这个过程是不是和数据库里查询一个人的名字很象?当然,xsl:value-of查询还有更多,更复杂的语法,因为是涉及寻找和定位的功能,我们会放在后面的XPath语法中在仔细讲解。
同样功能的还有xsl:copy-of,用法一样,就不重复解释了。

xsl:for-each 

xsl:for-each语法允许你循环处理被选择的节点。例如:有一个含多个个人资料的XML文档: 

<?xml version="1.0" encoding="iso-8859-1"?>
<PEOPLE>
<PERSON>
<name>ajie</name>
<age>28</age>
</PERSON>
<PERSON>
<name>tom</name>
<age>24</age>
</PERSON>

..............
</PEOPLE> 

我需要显示所有人的姓名,则可以将XSLT代码写成:

<xsl:template match="PEOPLE">
<xsl:for-each select="child::PERSON">
<xsl:value-of select="name"/>
</ xsl:for-each>
</xsl:template>

 

xsl:if 

xsl:if类似普通程序语言的if条件语句,允许设定节点满足某个条件时,被模板处理。xsl:if的语法格式为: 

<xsl:if test=布尔表达式>
template body
</xsl:if> 

如下

<xsl:template match="PEOPLE">
<xsl:if test="@name">
<p><xsl:value-of select="@name"/></p>
</xsl:if>
</xsl:template> 

这段代码的意思是检测PEOPLE节点下所有的元素,如果发现有<name>元素,则将<name>元素的值输出。其中@符号是统配符,表示节点下所有的元素。

xsl:choose, xsl:when 和 xsl:otherwise 

xsl:if语法没有else的属性。如果我们要进行多项选择,那么就要使用xsl:choose / xsl:when / xsl:otherwise系列流程控制语法了。具体的使用请看下面的XSL文件例子: 

<xsl:template match="PEOPLE">
<xsl:choose>
<xsl:when test="@name = 'ajie'">
<B><xsl:value-of select="@name"/></B>
</xsl:when>
<xsl:when test="@name">
<I><xsl:value-of select="@name"/></I>
</xsl:when>
<xsl:otherwise>
No name available
</xsl:otherwise>
<xsl:choose>
</xsl:template> 

说明:首先在PEOPLE节点下寻找<name>属性值为ajie的元素,如果找到,将ajie用粗体输出;如果没有发现值为ajie的<name>元素,则将所有的<name>元素的值都用斜体输出;如果没有发现任何<name>元素,则显示"No name available"。 

xsl:sort 

在XSLT中可以对XML源文档的元素进行重新排序,排序的语法就是xsl:sort。举例:下面的代码就是将文档元素按name排序。 

<xsl:template match="PEOPLE">
<xsl:apply-templates select="PERSON">
<xsl:sort select="@name"/>
</xsl:apply-templates>
</xsl:template> 

 

XPath的语法

1.当前位置
当我们使用XSLT处理XML源文档是,我们用Context来表示当前正在被模板处理的节点位置。比如xsl:template match="/"语句中表示Context在文档的根(root)节点。我不知道如何准确的翻译Context这个词,它类似于C语言里的指针,表示程序当前运行的位置。理解Context对于正确处理XSL模板非常重要,当您的XSL模板输出的文档和您想要的不一样,最先应该分析的就是Context在哪里。
Location Paths是用于设定你想要寻找的Context节点位置。就类似DOS的目录命令。我们看个例子
<xsl:for-each select="child::PEOPLE/descendant::PERSON">
其中child::PEOPLE/descendant::PERSON就是XPath语法,这个表达式就是一个Location Paths,代码说明要显示所有PEOPLE元素的子元素和所有PERSON元素的子元素。通常我们会采用更简单的写法:
<xsl:for-each select="PEOPLE//PERSON">
我们来解释path的两种表示方法:"/"和"//"。
"/"是表示当前文档的节点,类似DOS目录分割符。例如:/PEOPLE表示选择根节点下的PEOPLE元素;PEOPLE/PERSON表示选择PEOPLE元素下所有的PESON子元素。
"//"则表示当前文档所有的节点。类似查看整个目录。例如://PEOPLE表示选择文档中所有的PEOPLE元素,无论它在什么层次;PEOPLE//PERSON表示在PEOPLE元素下所有的PERSON元素,无论它的层次多深。

2 寻址操作
Axis和Predicate是XPath语法中对Location Paths进行定位操作的语法,具体的用法列表如下
Axis语法表
--------------------------------------------------------
表达式 简写 说明
--------------------------------------------------------
self . 选择当前的节点.。
例子 :
<TD><xsl:value-of select="."/></TD>
代码表示在当前位置插入当前的节点包含的文本(text)值,
--------------------------------------------------------
parent .. 选择当前节点的父节点。
--------------------------------------------------------
attribute @ 选择一个元素的所有属性。
例子:
<TD><xsl:value-of select="@PERSONID"/></TD>
选择PERSON元素的所有属性.
--------------------------------------------------------
child 选择当前节点的所有子元素。
--------------------------------------------------------
ancestor 选择当前节点的所有父元素(包括父元素的父元素,类推)
--------------------------------------------------------
Axis帮助我们选择当前节点周围所有的节点,而Predicate则用来定位当前节点内部的元素。表示方法为方括号[]中加表达式:[ Expression ]。具体举例如下:
PERSON[position()=2] 这句代码表示寻找第二个"PERSON" 元素
PERSON[starts-with(name, "B")] 这句代码表示寻找所有名称以"B"开头的PERSON元素。
3 运算符
这一节介绍XPath的运算符(Expressions),列表如下:
--------------------------------------------------------
运算符 说明
--------------------------------------------------------
and, or 就是普通意义的and, or
--------------------------------------------------------
= 等于
--------------------------------------------------------
!= 不等于
--------------------------------------------------------
>, >= 大于,大于等于
--------------------------------------------------------
<, <= 小于,小于等于。注意:在XSL文件中,<符号要用< 表示
--------------------------------------------------------
+, -, *, div 加减乘除
--------------------------------------------------------
mod 取模
--------------------------------------------------------
两个节点一起计算
--------------------------------------------------------
4 功能函数(Functions)
在XPath里有很多功能函数可以帮助我们精确寻找需要的节点。
count()功能
作用:统计计数,返回符合条件的节点的个数。
举例:<p><xsl:value-of select="count(PERSON[name=tom])"/></p>
说明:代码的用途是显示PERSON元素中姓名属性值为tom有几个。
number()功能
作用:将属性的值中的文本转换为数值。
举例:<p>The number is: <xsl:value-of select="number(book/price)"/></p>
说明:代码的用途是显示书的价格。
substring() 功能
语法:substring(value, start, length)
作用:截取字符串。
举例:<p><xsl:value-of select="substring(name, 1, 3)"/></p>
说明:代码的用途是截取name元素的值,从第一个字母开始显示到第三个。
sum()功能
作用:求和。
举例:<p>Total Price = <xsl:value-of select="sum(//price)"/></p>
说明:代码的用途是计算所有价格的和。
上面这些功能只是XPath语法中的一部分,还有大量的功能函数没有介绍,而且目前XPath的语法仍然在不断发展中。通过这些函数我们可以实现更加复杂的查询和操作。

XSLT是一种转换XML文档的语言,它包含两个过程:转换和格式化。XSLT的功能比CSS强大得多,它有类似数据查询的语法。

import, include, element, attribute, number, param等等语法,在这里就不一一解释。我们的目的是让您对XSLT的语法有基本的概念,理解XSLT作为一种转换语言的强大功能。

 

有关学习网站

◇ 最权威的网站
http://www.w3.org/Style/XSL/
◇ 有关标准
XSLT1.0 http://www.w3.org/TR/xslt.html
XSLT1.1 http://www.w3.org/TR/xslt11/
XPath1.0 http://www.w3.org/TR/xpath.html
◇ 学习教程
http://www.w3schools.com/xsl/
http://www.wirelessdevnet.com/channels/wap/training/xslt.html
http://msdn.microsoft.com/library/default.asp?URL=/library/psdk/xmlsdk/xslp8tlx.htm
http://www.jenitennison.com/xslt/
http://www.arbortext.com/xsl/
◇ 相关资讯
http://www.xslinfo.com/
http://www.xslt.com/
http://www.xml.com
http://www.oasis-open.org/cover/xsl.html#resources
http://www.xml.org
http://www.ibm.com/developer/xml
http://www.biztalk.org
◇ 相关软件
http://www.xmlsoftware.com/xsl/
◇ 邮件列表
http://www.15seconds.com/faq/XML/748.htm
http://www.mulberrytech.com/xsl/xsl-list/index.html