無(wú)為

          無(wú)為則可為,無(wú)為則至深!

            BlogJava :: 首頁(yè) :: 聯(lián)系 :: 聚合  :: 管理
            190 Posts :: 291 Stories :: 258 Comments :: 0 Trackbacks

          XML(可擴(kuò)展標(biāo)記語(yǔ)言)看起來(lái)可能像某種W3C標(biāo)準(zhǔn)——現(xiàn)在沒(méi)有什么實(shí)際影響,即使以后能派上用場(chǎng),也是很久以后的事。但實(shí)際上,它現(xiàn)在已經(jīng)得到了應(yīng)用。所以,不要等到XML已被加進(jìn)了你最喜愛(ài)的HTML編輯器中才開始使用它。它現(xiàn)在就可以解決各種內(nèi)部問(wèn)題和B2B系統(tǒng)問(wèn)題。
          在Sparks.com,我們使用XML來(lái)標(biāo)準(zhǔn)化從Java對(duì)象到HTML數(shù)據(jù)顯示等不同系統(tǒng)之間的數(shù)據(jù)表示。

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

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

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

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

          生成HTML
          XML加上XSL就等于HTML。這聽起來(lái)似乎不對(duì),但用戶所見(jiàn)的我們的HTML頁(yè)面其實(shí)就是XML和XSL共同產(chǎn)生的效果。


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

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

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

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

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

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

          是否使用DTD
          在Sparks.com,我們使用組織良好的XML,而不使用僅僅是正確的XML,因?yàn)榍罢卟恍枰狣TD。DTD在用戶點(diǎn)擊和看到頁(yè)面之間加入了一個(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ā)送給引擎。接下來(lái),事件與輸出文件同步。DOM解析引擎為動(dòng)態(tài)XML數(shù)據(jù)和XSL樣式表建立層次樹狀結(jié)構(gòu)。通過(guò)隨機(jī)訪問(wèn)DOM樹,可以提供XML數(shù)據(jù),就象由XSL樣式表來(lái)決定一樣。SAX模型上的爭(zhēng)論主要集中于對(duì)DOM結(jié)構(gòu)的內(nèi)存降低過(guò)度和加快XSL樣式表解析時(shí)間縮短方面。

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

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

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

          生成XML文檔相對(duì)來(lái)說(shuō)比較簡(jiǎn)單,因?yàn)樗恍枰粋€(gè)可以處理字符串的系統(tǒng)。我們建立了一個(gè)使用Java Servlet、Enterprise JavaBean server、JDBC和RDBMS(關(guān)系型數(shù)據(jù)庫(kù)管理系統(tǒng))的系統(tǒng)。

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

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

          啟動(dòng)XML生成過(guò)程的代碼放在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類來(lái)產(chǎn)生XML文件頭。我們的Java Servlet中的代碼如下:

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

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

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

          填入產(chǎn)品信息
          完成了文件頭后,控制方法會(huì)調(diào)用Java對(duì)象來(lái)產(chǎn)生它的XML。本例中調(diào)用的是product對(duì)象。product對(duì)象使用兩個(gè)方法來(lái)產(chǎn)生它的XML表示。第一個(gè)方法toXML()通過(guò)產(chǎn)生<product>和</product>標(biāo)簽來(lái)建立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> ");
          xml.append(internalXml());
          xml.append("</product> ");
          return xml.toString();
          }

          public String internalXml()
          {
          StringBuffer xml = new
          StringBuffer(" ")
          .append(productType).append(" ");
          xml.append(" ").append(idValue.trim())
          .append(" ");
          xml.append(" ").append(idName.trim())
          .append(" ");
          xml.append(" ").append(page.trim())
          .append(" ");
          厖?
          xml.append(" ").append(amount).append(" ");
          xml.append(" ").append(vendor).append(" ");
          xml.append(" ");
          xml.append(" ").append(pubDesc).append(" ");
          xml.append(" ").append(venDesc).append(" ";
          厖?
          return xml.toString();
          }


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

          三、用XSL作為模板語(yǔ)言

          為了得到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模板可以訪問(wèn)嵌在<product>標(biāo)簽內(nèi)的產(chǎn)品信息。我們又一次必須在文件末尾關(guān)閉<xsl:template>標(biāo)簽。

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


          <xsl:stylesheet xmlns:xsl="version="1.0">
          <xsl:template match="basketPage">
          <html>
          <head>
          <title>Shopping Bag / Adjust Quantity</title>
          </head>
          <body bgcolor="


          </xsl:template>
          </xsl:stylesheet>


          在模板的主體內(nèi),有很多XSL標(biāo)簽被用于為數(shù)據(jù)表示提供邏輯。下面解釋兩個(gè)常用的標(biāo)簽。
          Choose
          <xsl:choose>標(biāo)簽類似于傳統(tǒng)編程語(yǔ)言中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>來(lái)結(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="src="
          >
          <td valign="top" bgcolor="<font face="Verdana, Arial" size="1" color="<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)簽可以用來(lái)對(duì)相似XML數(shù)據(jù)的多種情況應(yīng)用同一個(gè)樣式表。對(duì)于我們來(lái)說(shuō),可以從數(shù)據(jù)庫(kù)中取出一系列產(chǎn)品信息,并在Web頁(yè)上進(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

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

          在Sparks.com,我們已經(jīng)在Java servlet里集成了一個(gè)XML解析器。這個(gè)解析器使用一種稱為XSLT (XSL Transformation)的機(jī)制,按XSL標(biāo)簽的說(shuō)明向XSL模板中添加X(jué)ML數(shù)據(jù)。

          當(dāng)我們的Java servlet處理HTTP請(qǐng)求時(shí),servlet檢索動(dòng)態(tài)生成的XML,然后XML被傳給解析引擎。根據(jù)XML文件中的指令,解析引擎查找適當(dāng)?shù)腦SL樣式表。解析器通過(guò)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ù)。
          ?
          ?



          凡是有該標(biāo)志的文章,都是該blog博主Caoer(草兒)原創(chuàng),凡是索引、收藏
          、轉(zhuǎn)載請(qǐng)注明來(lái)處和原文作者。非常感謝。

          主站蜘蛛池模板: 耒阳市| 岳普湖县| 巴中市| 类乌齐县| 巴里| 东丽区| 岳普湖县| 晴隆县| 翼城县| 随州市| 安泽县| 金湖县| 大邑县| 喀什市| 开鲁县| 屏南县| 承德市| 无极县| 富民县| 昭平县| 濮阳市| 万年县| 江油市| 集安市| 平罗县| 亚东县| 洛浦县| 万年县| 阿拉尔市| 宜丰县| 广平县| 云阳县| 牡丹江市| 库尔勒市| 全州县| 内丘县| 黑河市| 通江县| 涞源县| 奉化市| 汝州市|