MDA之路

          MDA,UML,XML,Eclipse及Java相關(guān)的Blog
          posts - 53, comments - 494, trackbacks - 0, articles - 2
            BlogJava :: 首頁 :: 新隨筆 :: 聯(lián)系 :: 聚合  :: 管理

          XSL:轉(zhuǎn)換從哪里開始?

          Posted on 2005-03-08 12:14 wxb_nudt 閱讀(4485) 評論(4)  編輯  收藏 所屬分類: 技術(shù)雜談

          XSL:轉(zhuǎn)換從哪里開始?

          前言

          愛好XML的人最終會試著將XML轉(zhuǎn)換HTML,或者轉(zhuǎn)換為其他類型的文檔,DOM/SAX顯然不是專門為轉(zhuǎn)換設(shè)計的,CSS對于轉(zhuǎn)換也是力有不逮,所以XML的愛好者們幾乎無一例外的要遭遇XSL,但是XSL似乎有非常多的用法,對于XML僅僅只是表示格式化的數(shù)據(jù)而言,XSL顯得復(fù)雜且毫無頭緒。

          例如《跟我學XSL》和《XSL基礎(chǔ)入門》這樣的教程會帶給你XSL的一些概念和例子,但是對于XSL的運行環(huán)境、平臺特性和本質(zhì),似乎都語焉不詳,你最終學會的僅僅是在XMLSPY或者IE中打開你的XML看看它轉(zhuǎn)換后的效果罷了。一有人提到腳本語言或者JAVA中調(diào)用XSL你就頭大了,甚至你不清楚XSLXSLT究竟有什么區(qū)別。迷失在網(wǎng)絡(luò)中的人們喜歡不停的用google搜索你想要的中文資料,但是其實有那個時間,干脆去那種技術(shù)的官方網(wǎng)站上好好看看吧。http://www.w3.org/Style/XSL/XSL技術(shù)的W3C的官方網(wǎng)站,在網(wǎng)頁正文的第一行它就解釋和XSLXSLT的區(qū)別。原文如下:

          XSL is a family of recommendations for defining XML document transformation and presentation. It consists of three parts:

          XSL Transformations (XSLT)

          a language for transforming XML

          the XML Path Language (XPath)

          an expression language used by XSLT to access or refer to parts of an XML document. (XPath is also used by the XML Linking specification)

          XSL Formatting Objects (XSL-FO)

          an XML vocabulary for specifying formatting semantics

          XSL是一組定義XML文檔的轉(zhuǎn)換和顯示特征的推薦標準,它包括三個部分:XSL轉(zhuǎn)換(XSLT)是一種為了轉(zhuǎn)換XML而定義的語言;XML路徑語言(XPath)是一種表達式語言,它被XSLT用來訪問或者提交一個XML文檔的某些部分(XPath也同時被XML Linking標準使用);XSL格式化對象(XSL-FO)是一個XML詞匯表用來定義XML的格式化語義。

          從何開始

          一般人學習XSL都是從XMLSPY等工具開始運行他的一個XSL例子,當然用文本編輯器編輯XMLXSL文件,用IE去打開XML也是一個好主意。因為XMLSPYIE都有嵌入式的XSL解析器,例如IEXSL解析器是MSXML,這樣不用顯式的調(diào)用XSL進行轉(zhuǎn)換過程,只需要在XML文檔的頭部加上一句<?xml:stylesheet type="text/xsl" href="xxx.xsl"?>就可以讓嵌入的XSL解析器自動的進行轉(zhuǎn)換了。例如下面這個著名的例子,它包括cd_catalog.xmlcd_catalog.xsl文件,內(nèi)容如下:

          xml文件:

          <?xml version="1.0" encoding="GB2312"?>

          <?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>

             <CD>

                 <TITLE>喀什噶爾胡楊</TITLE>

                 <ARTIST>刀郎</ARTIST>

                 <COUNTRY>China</COUNTRY>

                 <COMPANY>先之唱片</COMPANY>

                 <PRICE>20.60</PRICE>

                 <YEAR>2004</YEAR>

             </CD>

             <CD>

                 <TITLE>敦煌(特別版)</TITLE>

                 <ARTIST>女子十二樂坊</ARTIST>

                 <COUNTRY>China</COUNTRY>

                 <COMPANY>百代唱片</COMPANY>

                 <PRICE>25.60</PRICE>

                 <YEAR>2005</YEAR>

             </CD>

          </CATALOG>

           

          xsl文件:

          <?xml version="1.0" encoding="GB2312"?>

          <xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/1999/XSL/Transform" xmlns:fo="http://www.w3.org/1999/XSL/Format">

             <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>

          將它們保存在同一目錄下然后用IE5以上版本的IE直接打開xml文件,則會看到轉(zhuǎn)換后的效果。當然用XMLSPY中自帶的瀏覽器也可。

          JScript顯式調(diào)用XSL解析器

          上面的運行方法顯然是“貪天之功”,利用了IEXMLSPY自帶的XSL解析器,是讓一只看不見的手運行了轉(zhuǎn)換過程。那么,也可以用Jscript語言顯式的調(diào)用XSL解析器,讓沒有嵌入解析器的瀏覽器也可以運行XSL,當然,此瀏覽器必須支持Jscript腳本語言。我們還是使用上面的例子,不過將cd_catalog.xml中的<?xml:stylesheet type="text/xsl" href="cd_catalog.xsl"?>這一行去掉,同時新建一個cd_catalog.html文檔,內(nèi)容如下:

             <html>

              <body>

              <script language="javascript">

              // Load XML

              var xml = new ActiveXObject("Microsoft.XMLDOM")

              xml.async = false

              xml.load("cd_catalog.xml")

              // Load the XSL

              var xsl = new ActiveXObject("Microsoft.XMLDOM")

              xsl.async = false

              xsl.load("cd_catalog.xsl")

              // Transform

              document.write(xml.transformNode(xsl))

              </script>

              </body>

          </html>

          將此html文檔在支持Jscript的瀏覽器中打開,即可看到如前一段執(zhí)行的結(jié)果。當然不僅僅是Jscript,其他的腳本語言如VBScript等等也可以,不過JscriptXSL默認的腳本語言。

          腳本擴充的XSL,令人疑惑的xsl:eval標記

              xsl:eval標記并不是一個標準的xsl標記,它屬于http://www.w3.org/TR/WD-xsl這個名字空間,這個名字空間最終被微軟采用,于是xsl:eval也被微軟用來調(diào)用Jscript腳本,以此來擴充XSL的功能。而標準的XSL1.0版本的名字空間是http://www.w3.org/1999/XSL/Transform,它并不包含xsl:eval標記,這是很容易理解的,XSL應(yīng)該屬于一個平臺無關(guān)的技術(shù),如果它的某個標記要依賴微軟公司的產(chǎn)品,那顯然是自掘墳?zāi)埂jP(guān)于平臺無關(guān)的討論,將在本文的最后展開。

              xsl:eval標記的含義是計算其中腳本語言的表達式,并作為文本輸出。下面的例子中計算了cd_catalog.xml中各種CD的總價格,修改上面的cd_catalog.xsl并另存為cd_catalog2.xsl文件如下:

          <?xml version="1.0" encoding="GB2312"?>

          <xsl:stylesheet version="1.0" 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>

                           <tr>

                              <td>合計</td>

                              <td>

                                 <xsl:eval>total("PRICE")</xsl:eval>

                              </td>

                              <xsl:script>

                                 function total(q){

                                      temp=0;

                                      mark='/CATALOG/CD/'+q;

                                      v=selectNodes(mark);

                                      for(t=v.nextNode();t;t=v.nextNode()){

                                           temp+=Number(t.text);

                                      }

                                      return temp;

                                }

                              </xsl:script>

                           </tr>

                       </table>

                    </body>

                 </html>

             </xsl:template>

          </xsl:stylesheet>

          IE中打開cd_catalog.xml文件(注意修改xslcd_catalog2.xsl)即可看到結(jié)果,注意這個xsl文件的這一行<xsl:stylesheet version="1.0" xmlns:xsl="http://www.w3.org/TR/WD-xsl">,寫錯了名字空間xsl:eval標記就會報錯。

          瀏覽器無關(guān)的XSL解決方案,服務(wù)端的XSL

          不管如何折騰,要將XML通過XSL轉(zhuǎn)換為HTML必須要求本地主機上有一個XSL解析器,不管是瀏覽器內(nèi)嵌的,還是可以通過腳本語言調(diào)用。那么,更好的解決方案當然是從服務(wù)器端直接發(fā)送HTML回來,這樣無論什么瀏覽器都可以看到轉(zhuǎn)換的結(jié)果了。ASP提供了這個功能,這是可想而知的,不過我對ASP不熟,這段略過,有興趣的可以找本ASPXML教材看看。

          應(yīng)用程序中的XSL,語言相關(guān)的XSL

          眾所周知,Java是對XML技術(shù)支持得最好的語言,Java上面的xml包非常多,其中支持XSL轉(zhuǎn)換的包最著名的有SaxonxalanSaxon包可以在http://saxon.sourceforge.net/上面下載。將Saxon包解壓縮到C:\saxon6_5_36.5.3版本提供了對XSL1.0最穩(wěn)定的支持。然后在Classpath中加入C:\saxon6_5_3\saxon.jar;C:\saxon6_5_3\saxon-jdom.jar

          Saxon提供命令行式的XSL轉(zhuǎn)換和API。其中命令行式的轉(zhuǎn)換如下,將目錄移動到存放xml(去掉xml的指定xsl的那一行)和xsl的目錄,然后輸入下面的命令:

          java com.icl.saxon.StyleSheet cd_catalog.xml cd_catalog.xsl

          就可以看到輸出在屏幕上的結(jié)果,但是這樣看起來不方便,所以輸入如下命令:

          java com.icl.saxon.StyleSheet cd_catalog.xml cd_catalog.xsl>a.html

          然后將生成的a.html在瀏覽器中打開,可以清晰的看到結(jié)果。

          下面是在Java程序中調(diào)用Saxon包,進行XSL轉(zhuǎn)換的例子,文件名為XslExam.java

          import java.io.File;

          import javax.xml.transform.Result;

          import javax.xml.transform.Source;

          import javax.xml.transform.Templates;

          import javax.xml.transform.Transformer;

          import javax.xml.transform.TransformerConfigurationException;

          import javax.xml.transform.TransformerException;

          import javax.xml.transform.sax.SAXSource;

          import javax.xml.transform.stream.StreamResult;

          import com.icl.saxon.ExtendedInputSource;

          import com.icl.saxon.TransformerFactoryImpl;

           

          public class XSLExam {

                public static void main(String[] args) {

                  String sourceFileName = "cd_catalog.xml";

                  String styleFileName = "cd_catalog.xsl";

                  String outputFileName = "result.html";

                  File sourceFile = null;

                  File styleFile = null;

                  File outputFile = null;

                 

                  TransformerFactoryImpl factory = new TransformerFactoryImpl();

                 

                  Source sourceInput = null;

                  sourceFile = new File(sourceFileName);

                  ExtendedInputSource eis = new ExtendedInputSource(sourceFile);

                  sourceInput = new SAXSource(factory.getSourceParser(), eis);

                  eis.setEstimatedLength((int)sourceFile.length());

                 

                  Source styleSource ;

                  File sheetFile = new File(styleFileName);

                  eis = new ExtendedInputSource(sheetFile);

                  styleSource = new SAXSource(factory.getStyleParser(), eis);

                 

                  outputFile=new File(outputFileName);

                 

                       try {

                               Templates sheet = factory.newTemplates(styleSource);

                               Transformer instance = sheet.newTransformer();

                         Result result = new StreamResult(outputFile);

                         instance.transform(sourceInput, result);

                        

                        } catch (TransformerConfigurationException e) {

                               e.printStackTrace();

                        }catch (TransformerException err) {

                               err.printStackTrace();

                  }

                 }

          }

          這個例子程序?qū)?/SPAN>cd_catalog.xml文件使用cd_catalog.xsl轉(zhuǎn)換為result.html。在Eclipse3.01中調(diào)試通過(Saxon沒有簡單的xsl示例程序,我也是將com.icl.saxon.StyleSheet類拔光了才得到這個稍微簡單的例子,如果需要更詳細的用法,參考com.icl.saxon.StyleSheet類)。

          數(shù)據(jù)是獨立的,處理是平臺相關(guān)的

              總結(jié)前面的內(nèi)容,可以看出XSL轉(zhuǎn)換可以從這幾個地方開始:

          ?         IEXMLSPY:嵌入的解析器,例如MSXML3

          ?         JScript,顯式調(diào)用XSL解析器;

          ?         JScript擴充XSL功能,半吊子的XSL

          ?         瀏覽器無關(guān)的XSL解決方案,服務(wù)器端的XSLASP顯式調(diào)用XSL

          ?         語言相關(guān)的XSLJavaXSLSaxonxalan

          可以看出來,XSL無論如何,都是要平臺相關(guān)的,第一種方法依賴嵌入瀏覽器的XSL解析器;第二、三種方法依賴操作系統(tǒng)安裝的XSL解析器;第四種方法依賴服務(wù)器端安裝的XSL解析器;最后的方法依賴JAVA語言提供的XSL API。其中微軟還不顧W3C的反對,自定義了XSL的腳本擴充功能,功能倒是強大了,可惜脫離了Windows就玩不轉(zhuǎn)了。JAVA號稱平臺無關(guān),可是JAVA本身就是一個平臺,要是有人的機器沒有JRE又怎么辦呢?丟棄XSL

          不過事物總是有因果的,其實XML作為數(shù)據(jù)的存儲載體,可以做到完全的平臺無關(guān),但是XSL作為一個可執(zhí)行的語言,一定要依賴某種已存在的運行環(huán)境的,就如同數(shù)據(jù)庫中的表格和SQL語言一樣。SQL號稱適用于任何關(guān)系數(shù)據(jù)庫,但是實際上還是需要一個環(huán)境來run的。那么XSL是否破壞了XML的平臺無關(guān)性呢?我認為沒有,因為XSL本身是一個XML文檔,XML文檔可以平臺無關(guān)的保存和傳輸,至于使用何種方法來調(diào)用它則是另外考慮的問題。再者,XSL的源和目標都是平臺無關(guān)的文檔(例如XMLHTML),而它自己的調(diào)用方式則是可替換的,這點也減輕了XSL的負罪感吧。

          以上的討論都是基于XSL1.0標準的,目前XSL2.0標準尚在討論中,不過初稿已經(jīng)發(fā)布了,而Saxon8.0以上的版本號稱已經(jīng)支持了XSL2.0。讓我們拭目以待XSL2.0帶給我們的驚喜。

          參考文獻

          W3C站點:http://www.w3.org/Style/XSL/

          XSL主題:http://www-900.ibm.com/developerWorks/cn/xml/theme/x-xsl.shtml

          中文譯文站點:http://www.opendl.com/

          XSLT是什么類型的語言,SAXON的作者談XSLhttp://www-900.ibm.com/developerWorks/cn/xml/x-xslt/index.shtml

           

          注意:請轉(zhuǎn)載我文章的朋友,加上原文地址!!!上次我的那篇《XML本質(zhì)討論》很快就被轉(zhuǎn)載了,但是沒有聲明從這兒轉(zhuǎn)載。


          評論

          # re: XSL:轉(zhuǎn)換從哪里開始?  回復(fù)  更多評論   

          2005-03-08 12:38 by BJ
          頂,好東西!~
          收藏ing.....

          # re: XSL:轉(zhuǎn)換從哪里開始?  回復(fù)  更多評論   

          2005-03-10 16:21 by Boris-Java
          不知道你對cocoon了解多少,我最近也在研究xml和java,有空來我家坐坐?大家交流一下
          http://www.aygfsteel.com/lqsun/archive/2005/03/10/1911.html

          # re: XSL:轉(zhuǎn)換從哪里開始?  回復(fù)  更多評論   

          2005-06-19 18:15 by 小莫
          頂。很合我看呢。

          # re: XSL:轉(zhuǎn)換從哪里開始?  回復(fù)  更多評論   

          2006-05-24 15:03 by 周智理
          幫了我的大忙!謝謝
          主站蜘蛛池模板: 岫岩| 宜宾县| 射洪县| 拉萨市| 安宁市| 偃师市| 合作市| 广饶县| 永善县| 比如县| 绥芬河市| 合肥市| 衡阳县| 汝州市| 怀集县| 当阳市| 新昌县| 衡山县| 周宁县| 喜德县| 堆龙德庆县| 新晃| 永寿县| 山东省| 尉犁县| 大英县| 闵行区| 宽甸| 湘潭市| 密云县| 隆昌县| 星座| 玉山县| 汾阳市| 双牌县| 囊谦县| 抚远县| 新龙县| 格尔木市| 辽阳市| 嘉鱼县|