Sparks.com用XML和XSL來產生動態網頁面

來源:互聯網
上載者:User
xml|動態|頁面 XML(可延伸標記語言 (XML))看起來可能像某種W3C標準——現在沒有什麼實際影響,即使以後能派上用場,也是很久以後的事。但實際上,它現在已經得到了應用。所以,不要等到XML已被加進了你最喜愛的HTML編輯器中才開始使用它。它現在就可以解決各種內部問題和B2B系統問題。

在Sparks.com,我們使用XML來標準化從Java對象到HTML資料顯示等不同系統之間的資料表示。

特別需要指出的是,我們發現,只要以非常基本的XML結構來實現標準化,就可以更容易地共用和操作資料。在這一過程中,我們發現了使用XML的很多有效方法。下面詳細介紹我們現在的應用情況。

標準化
在使用XML之前,建立與你要使用的資訊不同的XML資料格式。

產生動態XML
從資料庫中產生HTML並不新鮮,但產生XML卻很新鮮。這裡我們介紹具體的產生步驟。

用XSL作為範本語言
XSL(可延伸樣式表語言 (XSL))是定義XML資料顯示格式的好方法,如果寫成幾個靜態模板會更有效。

產生HTML
XML加上XSL就等於HTML。這聽起來似乎不對,但使用者所見的我們的HTML頁面其實就是XML和XSL共同產生的效果。


一、標準化

XML的能力來自於它的靈活性。但不幸的是,它有時太靈活了,以至於你會面對一個空白的頁面,發愁該怎麼解決問題。

在任何XML的項目中,第一步工作都是建立標準的資料格式。為此你要作出以下決定:

• 要涉及哪些資料
• 是否要使用DTD(檔案類型定義)
• 是否要使用DOM(文件物件模型)或SAX(XML的簡化API)解析

確定資料:
因為沒有標準的XML格式,開發人員可以自由地開發自己的格式。然而,如果你的格式只能被一個應用程式識別,那麼你只能運行這個程式來使用該格式。如果還有其他程式也能讀懂你的XML格式,那顯然會更有協助。如果某個XML格式被修改,則使用它的系統可能也需要被修改,所以你應該建立儘可能完整的格式。因為大多數系統忽略它們無法識別的標籤,所以改變一個XML格式的最安全的方法是添加標籤,而不是修改標籤。

單擊此處查看XML資料格式執行個體

在Sparks.com,我們查看了不同的產品展示需要的所有產品資料。儘管並不是所有的頁面都使用全部資料,但我們還是由此開發出適用於所有資料的非常完整的XML資料格式。例如,我們的產品明細資訊頁面顯示的資料要比產品瀏覽頁面多。然而,我們在這兩種情況下仍然使用相同的資料格式,因為每個頁面的XSL模板都只使用它所需要的欄位。

是否使用DTD
在Sparks.com,我們使用組織良好的XML,而不使用僅僅是正確的XML,因為前者不需要DTD。DTD在使用者點擊和看到頁面之間加入了一個處理層。我們發現這一層需要太多的處理。當然,在以XML格式與其他公司通訊時,使用DTD還是很不錯的。因為DTD能在發送和接受時能保證資料結構正確。

選擇解析引擎
現在,可以使用的解析引擎有好幾個。選擇哪一個幾乎完全取決於你的應用需要。如果你決定使用DTD,那麼這個解析引擎必須能使你的XML被DTD驗證。你可以將驗證另放到一個進程中,但那樣會影響效能。

SAX和DOM是兩個基本的解析模型。SAX基於事件,所以在XML被解析時,事件被發送給引擎。接下來,事件與輸出檔案同步。DOM解析引擎為動態XML資料和XSL樣式表建立層次樹狀結構。通過隨機訪問DOM樹,可以提供XML資料,就象由XSL樣式表來決定一樣。SAX模型上的爭論主要集中於對DOM結構的記憶體降低過度和加快XSL樣式表解析時間縮短方面。

然而,我們發現使用SAX的很多系統並沒有充分發揮它的能力。這些系統用它來建立DOM結構並通過DOM結構來發送事件。用這種方法,在任何XML處理之前必須從樣式表中建立DOM,所以效能會下降。

二、產生動態XML

一旦建立了XML格式,我們需要一種能夠將其從資料庫中動態移植的方法。

產生XML文檔相對來說比較簡單,因為它只需要一個可以處理字串的系統。我們建立了一個使用Java Servlet、Enterprise JavaBean server、JDBC和RDBMS(關係型資料庫管理系統)的系統。

• Servlet通過把產生XML文檔的任務交給Enterprise JavaBean (EJB)來處理產品資訊請求。
• EJB使用JDBC從資料庫裡查詢所需的產品詳細資料。
• EJB產生XML檔案並把它傳遞給Servlet。
• Servlet調用解析引擎,從XML檔案和靜態XSL樣式表中建立HTML輸出。

(有關XSL應用的其他資訊,請參閱用XSL作為範本語言。)

產生XML的例子
在Java中建立XML文檔字串的真正代碼可以分成幾個方法和類。

啟動XML產生過程的代碼放在EJB方法裡。這一執行個體會立即建立一個StringBuffer,以便儲存產生的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());


後面的三個xml.append()變元本身就是對其他方法的調用。
產生檔案頭
第一個附加方法調用XmlUtils類來產生XML檔案頭。我們的Java Servlet中的代碼如下:

public static String beginDocument(String stylesheet, String page)
{
    StringBuffer xml = new StringBuffer();
    xml.append("<?xml version=\"1.0\"?>\n")
    .append("<?xml-stylesheet href=\"")
    .append(stylesheet).append("\"")
    .append(" type =\"text/xsl\"?>\n");
  xml.append("<").append(page).append(">\n");
  return xml.toString();
}

這段代碼產生了XML檔案頭。<?xml>標籤把本檔案定義為支援1.0版本的XML檔案。第二行代碼指向用以顯示資料的正確樣式表的位置。最後包括進去的是項級標籤(本執行個體中為<browse>)。在檔案末尾,只有<browse>標籤需要被關閉。

<?xml version="1.0"?> <?xml-stylesheet href="/browse_find/browse.xsl" type="text/xsl"?> <browse>

填入產品資訊
完成了檔案頭後,控制方法會調用Java對象來產生它的XML。本例中調用的是product對象。product對象使用兩個方法來產生它的XML表示。第一個方法toXML()通過產生<product>和</product>標籤來建立product節點。然後它會調用internalXML(),這樣就能提供產品XML所需的內容。internalXML()是一系列的StringBuffer.append()調用。StringBuffer也被轉換成字串並返回給控制方法。
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();
}


關閉檔案
最後,XMLUtils.endDocument()方法被調用。這個調用關閉XML標籤(本例中為),並最終完成架構好的XML檔案。來自控制方法的整個StringBuffer也轉換成字串,並返回給處理最初HTTP請求的servlet。

三、用XSL作為範本語言

為了得到HTML輸出,我們把產生的XML檔案和控制XML資料如何表示的XSL模板相結合。我們的XSL模板由精心組織的XSL和HTML標籤組成。

開始建模板
我們的XSL模板開始部分與下面這段代碼類似。第一行代碼為必需代碼,將本檔案定義為XSL樣式表。xmlns:xsl=屬性引用本檔案所使用的XML名稱空間,而version=屬性則定義名稱空間的版本號碼。在檔案的末尾,我們關閉標籤。

由<xsl:template>開始的第二行代碼確定了XSL模板的模式。Match屬性是必需的,在這裡指向XML標籤<basketPage>。在我們的系統裡,<basketPage>標籤裡包含<product> 標籤,這使得XSL模板可以訪問嵌在<product>標籤內的產品資訊。我們又一次必須在檔案末尾關閉<xsl:template>標籤。

接下來,我們來看一看組織良好的HTML。由於它將被XML解析引擎處理,所以必須符合組織良好的XML的所有規則。從本質上來講,這意味著所有的開始標籤必須有對應的結束標籤。例如,通常不被結束的<P>標籤,必須用</P>關閉。


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


在模板的主體內,有很多XSL標籤被用於為資料表示提供邏輯。下面解釋兩個常用的標籤。
Choose
<xsl:choose>標籤類似於傳統程式設計語言中if-then-else結構的開始部分。在XSL中,choose標籤表示在代碼進入的部分中,賦值將觸發動作的發生。擁有賦值屬性的<xsl:when>標籤跟在choose標籤後面。如果賦值是正確的,位於<xsl:when>的開始和結束標籤之間的內容將被使用。如果賦值錯誤,就使用<xsl:otherwise>的開始和結束標籤之間的內容。整個部分用</xsl:choose>來結束。

在這個例子裡,when標籤會為quantity標籤檢查XML。如果quantity標籤裡含有值為真的error屬性,quantity標籤將會顯示列在下面的表格單元。如果屬性的值不為真,XSL將會顯示otherwise標籤間的內容。在下面的執行個體裡,如果error屬性不真,則什麼都不會被顯示。

<xsl:choose>
<xsl:when test="quantity[@error='true']">
<td bgcolor="#ffffff"><img height="1" width="1" src="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>標籤可以用來對相似XML資料的多種情況應用同一個樣式表。對於我們來說,可以從資料庫中取出一系列產品資訊,並在Web頁上進行統一格式化。這裡有一個例子:
<xsl:for-each select="package">
<xsl:apply-templates select="product"/>
</xsl:for-each>


for-each 迴圈在程式遇到標籤時開始。這個迴圈將在程式遇到標籤時結束。一旦這個迴圈運行,每次標籤出現時都會應用這個模板。

四、產生HTML

將來的某一時刻,瀏覽器將會整合XML解析引擎。到那時,你可以直接向瀏覽器發送XML和XSL檔案,而瀏覽器則根據樣式表中列出的規則顯示XML資料。不過,在此之前開發人員們將不得不在他們伺服器端的系統裡建立解析功能。

在Sparks.com,我們已經在Java servlet裡整合了一個XML解析器。這個解析器使用一種稱為XSLT (XSL Transformation)的機制,按XSL標籤的說明向XSL模板中添加XML資料。

當我們的Java servlet處理HTTP請求時,servlet檢索動態產生的XML,然後XML被傳給解析引擎。根據XML檔案中的指令,解析引擎尋找適當的XSL樣式表。解析器通過DOM結構建立HTML檔案,然後這個檔案再傳送給發出HTTP請求的使用者。

如果你選擇使用SAX模型,解析器會通讀XML來源程式,為每個XML標籤建立一個事件。事件與XML資料對應,並最終按XSL標籤向樣式表中插入資料。



相關文章

Beyond APAC's No.1 Cloud

19.6% IaaS Market Share in Asia Pacific - Gartner IT Service report, 2018

Learn more >

Apsara Conference 2019

The Rise of Data Intelligence, September 25th - 27th, Hangzhou, China

Learn more >

Alibaba Cloud Free Trial

Learn and experience the power of Alibaba Cloud with a free trial worth $300-1200 USD

Learn more >

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。