明輝手游網(wǎng)中心:是一個(gè)免費(fèi)提供流行視頻軟件教程、在線學(xué)習(xí)分享的學(xué)習(xí)平臺(tái)!

用XML與XSL來生成動(dòng)態(tài)頁面

[摘要]xml(可擴(kuò)展標(biāo)記語言)看起來可能像某種w3c標(biāo)準(zhǔn)——現(xiàn)在沒有什么實(shí)際影響,即使以后能派上用場(chǎng),也是很久以后的事。但實(shí)際上,它現(xiàn)在已經(jīng)得到了應(yīng)用。所以,不要等到xml已被加進(jìn)了你最喜愛的html編輯器中才開始使用它。它現(xiàn)在就可以解決各種內(nèi)部問題和b2b系統(tǒng)問題。 在sparks.com,我們使用x...

xml(可擴(kuò)展標(biāo)記語言)看起來可能像某種w3c標(biāo)準(zhǔn)——現(xiàn)在沒有什么實(shí)際影響,即使以后能派上用場(chǎng),也是很久以后的事。但實(shí)際上,它現(xiàn)在已經(jīng)得到了應(yīng)用。所以,不要等到xml已被加進(jìn)了你最喜愛的html編輯器中才開始使用它。它現(xiàn)在就可以解決各種內(nèi)部問題和b2b系統(tǒng)問題。

在sparks.com,我們使用xml來標(biāo)準(zhǔn)化從java對(duì)象到html數(shù)據(jù)顯示等不同系統(tǒng)之間的數(shù)據(jù)表示。

特別需要指出的是,我們發(fā)現(xiàn),只要以非常基本的xml結(jié)構(gòu)來實(shí)現(xiàn)標(biāo)準(zhǔn)化,就可以更容易地共享和操作數(shù)據(jù)。在這一過程中,我們發(fā)現(xiàn)了使用xml的很多有效方法。下面詳細(xì)介紹我們現(xiàn)在的應(yīng)用情況。

標(biāo)準(zhǔn)化
在使用xml之前,建立與你要使用的信息不同的xml數(shù)據(jù)格式。

生成動(dòng)態(tài)xml
從數(shù)據(jù)庫中生成html并不新鮮,但生成xml卻很新鮮。這里我們介紹具體的生成步驟。

用xsl作為模板語言
xsl(可擴(kuò)展樣式表語言)是定義xml數(shù)據(jù)顯示格式的好方法,如果寫成幾個(gè)靜態(tài)模板會(huì)更有效。

生成html
xml加上xsl就等于html。這聽起來似乎不對(duì),但用戶所見的我們的html頁面其實(shí)就是xml和xsl共同產(chǎn)生的效果。


一、標(biāo)準(zhǔn)化

xml的能力來自于它的靈活性。但不幸的是,它有時(shí)太靈活了,以至于你會(huì)面對(duì)一個(gè)空白的頁面,發(fā)愁該怎么解決問題。

在任何xml的項(xiàng)目中,第一步工作都是創(chuàng)建標(biāo)準(zhǔn)的數(shù)據(jù)格式。為此你要作出以下決定:

• 要涉及哪些數(shù)據(jù)
• 是否要使用dtd(文件類型定義)
• 是否要使用dom(文檔對(duì)象模型)或sax(xml的簡化api)解析

確定數(shù)據(jù):
因?yàn)闆]有標(biāo)準(zhǔn)的xml格式,開發(fā)者可以自由地開發(fā)自己的格式。然而,如果你的格式只能被一個(gè)應(yīng)用程序識(shí)別,那么你只能運(yùn)行這個(gè)程序來使用該格式。如果還有其他程序也能讀懂你的xml格式,那顯然會(huì)更有幫助。如果某個(gè)xml格式被修改,則使用它的系統(tǒng)可能也需要被修改,所以你應(yīng)該建立盡可能完整的格式。因?yàn)榇蠖鄶?shù)系統(tǒng)忽略它們無法識(shí)別的標(biāo)簽,所以改變一個(gè)xml格式的最安全的方法是添加標(biāo)簽,而不是修改標(biāo)簽。

單擊此處查看xml數(shù)據(jù)格式實(shí)例

在sparks.com,我們查看了不同的產(chǎn)品展示需要的所有產(chǎn)品數(shù)據(jù)。盡管并不是所有的頁面都使用全部數(shù)據(jù),但我們還是由此開發(fā)出適用于所有數(shù)據(jù)的非常完整的xml數(shù)據(jù)格式。例如,我們的產(chǎn)品明細(xì)信息頁面顯示的數(shù)據(jù)要比產(chǎn)品瀏覽頁面多。然而,我們?cè)谶@兩種情況下仍然使用相同的數(shù)據(jù)格式,因?yàn)槊總(gè)頁面的xsl模板都只使用它所需要的字段。

是否使用dtd
在sparks.com,我們使用組織良好的xml,而不使用僅僅是正確的xml,因?yàn)榍罢卟恍枰猟td。dtd在用戶點(diǎn)擊和看到頁面之間加入了一個(gè)處理層。我們發(fā)現(xiàn)這一層需要太多的處理。當(dāng)然,在以xml格式與其他公司通信時(shí),使用dtd還是很不錯(cuò)的。因?yàn)閐td能在發(fā)送和接受時(shí)能保證數(shù)據(jù)結(jié)構(gòu)正確。

選擇解析引擎
現(xiàn)在,可以使用的解析引擎有好幾個(gè)。選擇哪一個(gè)幾乎完全取決于你的應(yīng)用需要。如果你決定使用dtd,那么這個(gè)解析引擎必須能使你的xml被dtd驗(yàn)證。你可以將驗(yàn)證另放到一個(gè)進(jìn)程中,但那樣會(huì)影響性能。

sax和dom是兩個(gè)基本的解析模型。sax基于事件,所以在xml被解析時(shí),事件被發(fā)送給引擎。接下來,事件與輸出文件同步。dom解析引擎為動(dòng)態(tài)xml數(shù)據(jù)和xsl樣式表建立層次樹狀結(jié)構(gòu)。通過隨機(jī)訪問dom樹,可以提供xml數(shù)據(jù),就象由xsl樣式表來決定一樣。sax模型上的爭論主要集中于對(duì)dom結(jié)構(gòu)的內(nèi)存降低過度和加快xsl樣式表解析時(shí)間縮短方面。

然而,我們發(fā)現(xiàn)使用sax的很多系統(tǒng)并沒有充分發(fā)揮它的能力。這些系統(tǒng)用它來建立dom結(jié)構(gòu)并通過dom結(jié)構(gòu)來發(fā)送事件。用這種方法,在任何xml處理之前必須從樣式表中建立dom,所以性能會(huì)下降。

二、生成動(dòng)態(tài)xml

一旦建立了xml格式,我們需要一種能夠?qū)⑵鋸臄?shù)據(jù)庫中動(dòng)態(tài)移植的方法。

生成xml文檔相對(duì)來說比較簡單,因?yàn)樗恍枰粋(gè)可以處理字符串的系統(tǒng)。我們建立了一個(gè)使用java servlet、enterprise javabean server、jdbc和rdbms(關(guān)系型數(shù)據(jù)庫管理系統(tǒng))的系統(tǒng)。

• servlet通過把生成xml文檔的任務(wù)交給enterprise javabean (ejb)來處理產(chǎn)品信息請(qǐng)求。
• ejb使用jdbc從數(shù)據(jù)庫里查詢所需的產(chǎn)品詳細(xì)信息。
• ejb生成xml文件并把它傳遞給servlet。
• servlet調(diào)用解析引擎,從xml文件和靜態(tài)的xsl樣式表中創(chuàng)建html輸出。

(有關(guān)xsl應(yīng)用的其他信息,請(qǐng)參閱用xsl作為模板語言。)

生成xml的例子
在java中創(chuàng)建xml文檔字符串的真正代碼可以分成幾個(gè)方法和類。

啟動(dòng)xml生成過程的代碼放在ejb方法里。這一實(shí)例會(huì)立即創(chuàng)建一個(gè)stringbuffer,以便存儲(chǔ)生成的xml字符串。

stringbuffer xml = new stringbuffer();
xml.append(xmlutils.begindocument("/browse_find/browse.xsl", "browse", request));
xml.append(product.toxml());
xml.append(xmlutils.enddocument("browse");
out.print(xml.tostring());


后面的三個(gè)xml.append()變?cè)旧砭褪菍?duì)其他方法的調(diào)用。
產(chǎn)生文件頭
第一個(gè)附加方法調(diào)用xmlutils類來產(chǎn)生xml文件頭。我們的java servlet中的代碼如下:

public static string begindocument(string stylesheet, string page)
{
    stringbuffer xml = new stringbuffer();
    xml.append("\n")
    .append("
    .append(stylesheet).append("\"")
    .append(" type =\"text/xsl\"?>\n");
  xml.append("<").append(page).append(">\n");
  return xml.tostring();
}

這段代碼生成了xml文件頭。標(biāo)簽把本文件定義為支持1.0版本的xml文件。第二行代碼指向用以顯示數(shù)據(jù)的正確樣式表的位置。最后包括進(jìn)去的是項(xiàng)級(jí)標(biāo)簽(本實(shí)例中為)。在文件末尾,只有標(biāo)簽需要被關(guān)閉。


[page_break]

填入產(chǎn)品信息
完成了文件頭后,控制方法會(huì)調(diào)用java對(duì)象來產(chǎn)生它的xml。本例中調(diào)用的是product對(duì)象。product對(duì)象使用兩個(gè)方法來產(chǎn)生它的xml表示。第一個(gè)方法toxml()通過產(chǎn)生<product>和</product>標(biāo)簽來建立product節(jié)點(diǎn)。然后它會(huì)調(diào)用internalxml(),這樣就能提供產(chǎn)品xml所需的內(nèi)容。internalxml()是一系列的stringbuffer.append()調(diào)用。stringbuffer也被轉(zhuǎn)換成字符串并返回給控制方法。
public string toxml()
    {
    stringbuffer xml = new stringbuffer("<product>\n");
    xml.append(internalxml());
    xml.append("</product>\n");
    return xml.tostring();
    }

public string internalxml()
    {
    stringbuffer xml = new
    stringbuffer("\t")
        .append(producttype).append("\n");
    xml.append("\t").append(idvalue.trim())
        .append("\n");
    xml.append("\t").append(idname.trim())
        .append("\n");
    xml.append("\t").append(page.trim())
        .append("\n");
厖?
      xml.append("\t").append(amount).append("\n");
    xml.append("\t").append(vendor).append("\n");
    xml.append("\t\n");
    xml.append("\t").append(pubdesc).append("\n");
    xml.append("\t").append(vendesc).append("\n";
厖?
    return xml.tostring();
}


關(guān)閉文件
最后,xmlutils.enddocument()方法被調(diào)用。這個(gè)調(diào)用關(guān)閉xml標(biāo)簽(本例中為),并最終完成架構(gòu)好的xml文件。來自控制方法的整個(gè)stringbuffer也轉(zhuǎn)換成字符串,并返回給處理最初http請(qǐng)求的servlet。

三、用xsl作為模板語言

為了得到html輸出,我們把生成的xml文件和控制xml數(shù)據(jù)如何表示的xsl模板相結(jié)合。我們的xsl模板由精心組織的xsl和html標(biāo)簽組成。

開始建模板
我們的xsl模板開始部分與下面這段代碼類似。第一行代碼為必需代碼,將本文件定義為xsl樣式表。xmlns:xsl=屬性引用本文件所使用的xml名稱空間,而version=屬性則定義名稱空間的版本號(hào)。在文件的末尾,我們關(guān)閉標(biāo)簽。

由<xsl:template>開始的第二行代碼確定了xsl模板的模式。match屬性是必需的,在這里指向xml標(biāo)簽<basketpage>。在我們的系統(tǒng)里,<basketpage>標(biāo)簽里包含<product> 標(biāo)簽,這使得xsl模板可以訪問嵌在<product>標(biāo)簽內(nèi)的產(chǎn)品信息。我們又一次必須在文件末尾關(guān)閉<xsl:template>標(biāo)簽。

接下來,我們來看一看組織良好的html。由于它將被xml解析引擎處理,所以必須符合組織良好的xml的所有規(guī)則。從本質(zhì)上來講,這意味著所有的開始標(biāo)簽必須有對(duì)應(yīng)的結(jié)束標(biāo)簽。例如,通常不被結(jié)束的<p>標(biāo)簽,必須用</p>關(guān)閉。


<xsl:stylesheet xmlns:xsl="http://www.w3.org/1999/xsl/transform"
version="1.0">
<xsl:template match="basketpage">
<html>
<head>
<title>shopping bag / adjust quantity</title>
</head>
<body bgcolor="#cccc99" bgproperties="fixed" link="#990000" vlink="#990000">
<br>
?br> </xsl:template>
</xsl:stylesheet>


在模板的主體內(nèi),有很多xsl標(biāo)簽被用于為數(shù)據(jù)表示提供邏輯。下面解釋兩個(gè)常用的標(biāo)簽。
choose
<xsl:choose>標(biāo)簽類似于傳統(tǒng)編程語言中if-then-else結(jié)構(gòu)的開始部分。在xsl中,choose標(biāo)簽表示在代碼進(jìn)入的部分中,賦值將觸發(fā)動(dòng)作的發(fā)生。擁有賦值屬性的<xsl:when>標(biāo)簽跟在choose標(biāo)簽后面。如果賦值是正確的,位于<xsl:when>的開始和結(jié)束標(biāo)簽之間的內(nèi)容將被使用。如果賦值錯(cuò)誤,就使用<xsl:otherwise>的開始和結(jié)束標(biāo)簽之間的內(nèi)容。整個(gè)部分用</xsl:choose>來結(jié)束。

在這個(gè)例子里,when標(biāo)簽會(huì)為quantity標(biāo)簽檢查xml。如果quantity標(biāo)簽里含有值為真的error屬性,quantity標(biāo)簽將會(huì)顯示列在下面的表格單元。如果屬性的值不為真,xsl將會(huì)顯示otherwise標(biāo)簽間的內(nèi)容。在下面的實(shí)例里,如果error屬性不真,則什么都不會(huì)被顯示。

<xsl:choose>
<xsl:when test="quantity[@error='true']">
<td bgcolor="#ffffff"><img height="1" width="1" src="http://img.sparks.com/images/i-catalog/sparks_images/sparks_ui/clearpixel.gif"/></td>
<td valign="top" bgcolor="#ffffff" colspan="2"><font face="verdana, arial" size="1" color="#cc3300"><b>*not enough in stock. your quantity was adjusted accordingly.</b></font></td>
</xsl:when>
<xsl:otherwise>
</xsl:otherwise>
</xsl:choose>


for-each
<xsl:for-each>標(biāo)簽可以用來對(duì)相似xml數(shù)據(jù)的多種情況應(yīng)用同一個(gè)樣式表。對(duì)于我們來說,可以從數(shù)據(jù)庫中取出一系列產(chǎn)品信息,并在web頁上進(jìn)行統(tǒng)一格式化。這里有一個(gè)例子:
<xsl:for-each select="package">
<xsl:apply-templates select="product"/>
</xsl:for-each>


for-each 循環(huán)在程序遇到標(biāo)簽時(shí)開始。這個(gè)循環(huán)將在程序遇到標(biāo)簽時(shí)結(jié)束。一旦這個(gè)循環(huán)運(yùn)行,每次標(biāo)簽出現(xiàn)時(shí)都會(huì)應(yīng)用這個(gè)模板。

四、生成html

將來的某一時(shí)刻,瀏覽器將會(huì)集成xml解析引擎。到那時(shí),你可以直接向?yàn)g覽器發(fā)送xml和xsl文件,而瀏覽器則根據(jù)樣式表中列出的規(guī)則顯示xml數(shù)據(jù)。不過,在此之前開發(fā)者們將不得不在他們服務(wù)器端的系統(tǒng)里創(chuàng)建解析功能。

在sparks.com,我們已經(jīng)在java servlet里集成了一個(gè)xml解析器。這個(gè)解析器使用一種稱為xslt (xsl transformation)的機(jī)制,按xsl標(biāo)簽的說明向xsl模板中添加xml數(shù)據(jù)。

當(dāng)我們的java servlet處理http請(qǐng)求時(shí),servlet檢索動(dòng)態(tài)生成的xml,然后xml被傳給解析引擎。根據(jù)xml文件中的指令,解析引擎查找適當(dāng)?shù)膞sl樣式表。解析器通過dom結(jié)構(gòu)創(chuàng)建html文件,然后這個(gè)文件再傳送給發(fā)出http請(qǐng)求的用戶。

如果你選擇使用sax模型,解析器會(huì)通讀xml源程序,為每個(gè)xml標(biāo)簽創(chuàng)建一個(gè)事件。事件與xml數(shù)據(jù)對(duì)應(yīng),并最終按xsl標(biāo)簽向樣式表中插入數(shù)據(jù)。