2020国产成人精品视频,性做久久久久久久久,亚洲国产成人久久综合一区,亚洲影院天堂中文av色

分享

集成JSP/JSF和XML/XSLT

 starsiday 2006-08-09
集成JSP/JSF和XML/XSLT
作者:Erik Bru… 文章來(lái)源:gceclub 點(diǎn)擊數(shù):193 更新時(shí)間:2006-6-4
【字體:

如果您有任何疑問(wèn),請(qǐng)到開(kāi)發(fā)論壇上提問(wèn)。

簡(jiǎn)介

大家期待已久的JavaServer FacesTM(JSF)版本1.0和JavaServer PagesTM(JSP)版本2.0終于發(fā)布,它們承諾要改變J2EE開(kāi)發(fā)人員構(gòu)建Web應(yīng)用程序的方式。與此同時(shí),可擴(kuò)展樣式表語(yǔ)言轉(zhuǎn)換(Extensible Stylesheet Language Transformation,XSLT)版本2.0已經(jīng)處于規(guī)范制定的最后階段,而許多開(kāi)發(fā)人員正在認(rèn)真地考慮基于XML的表示層,尤其是當(dāng)應(yīng)用程序以多種設(shè)備為目標(biāo),或者要求外觀定制時(shí)。

本文說(shuō)明了一點(diǎn),即JSF 1.0和JSP 2.0是邏輯演變的結(jié)果,而這種演變?cè)缭趕ervlet編程的早期就開(kāi)始了。借助諸如JSP 1.1、Struts 1.0和JSLT 1.0這些里程碑式的技術(shù),Java技術(shù)已經(jīng)逐漸集成了來(lái)自實(shí)際開(kāi)發(fā)領(lǐng)域的多種成功技術(shù)。接著,本文介紹了Model 2X,它結(jié)合了JSP技術(shù)和XML處理,是上述演變的延續(xù),并且增強(qiáng)了表示層的靈活性。

為了更好地理解本文中討論的概念,您應(yīng)該具有JSP、XML和Web應(yīng)用程序架構(gòu)方面的基本知識(shí)。與JSF概念的某些相似性將會(huì)大有幫助。

注意:書(shū)寫(xiě)本文過(guò)程中所使用的JSF規(guī)范的版本是JavaServer Faces 1.0 Specification Early Access Draft,版本日期為2002年9月6日。該規(guī)范將在未來(lái)不斷完善。所使用的JSF實(shí)現(xiàn)是Sun Reference implementation v1.0,Early Access 2(EA2)。

本文涉及到的所有源代碼都可以從下面的網(wǎng)址下載得到:
http://www./model2x/tss

對(duì)于Java開(kāi)發(fā)人員來(lái)說(shuō),下面列舉的都是激動(dòng)人心的時(shí)刻。在撰寫(xiě)本文的時(shí)候(2003年2月):

  • JavaServer Pages 2.0和Java ServletTM 2.4規(guī)范計(jì)劃定稿。
  • 自從2002年6月以來(lái),JavaServer Pages Standard Tag LibraryTM規(guī)范已經(jīng)定稿。
  • 獲得期望最高的JavaServer Faces 1.0正處于開(kāi)發(fā)階段且進(jìn)展順利。

即便您能夠做到與時(shí)俱進(jìn),規(guī)范和首字母縮寫(xiě)詞的豐富程度還是令人望而卻步。好消息是,所有這些技術(shù)都彼此相關(guān),且設(shè)計(jì)為彼此互補(bǔ)。

  • Java Servlet是Java Web技術(shù)的基礎(chǔ)。Java Servlet規(guī)范解決了如何打包和部署應(yīng)用程序的問(wèn)題,并描述了用于生成動(dòng)態(tài)內(nèi)容的服務(wù)器端組件(即servlet)。
  • JavaServer Pages(JSP)構(gòu)建在servlet技術(shù)之上,為生成諸如HTML這樣的文本內(nèi)容提供了一種頁(yè)面模板方法。
  • JavaServer Pages Standard Tag Library(JSTL)是一個(gè)標(biāo)準(zhǔn)JSP動(dòng)作(也稱為標(biāo)簽)的集合,它簡(jiǎn)化了JSP頁(yè)面的開(kāi)發(fā)工作。頁(yè)面作者可以使用JSTL構(gòu)建高級(jí)頁(yè)面邏輯,而不用了解Java或其他腳本語(yǔ)言。
  • JavaServer Faces(JSF)提供了一個(gè)可擴(kuò)展的、基于組件的、工具友好的服務(wù)器端UI框架,該框架可以與JSP及JSTL很好地集成。

下圖說(shuō)明了各種技術(shù)之間的相關(guān)性:

圖1:技術(shù)堆棧

從前…

在開(kāi)始實(shí)現(xiàn)JSP之前,服務(wù)器端的Java應(yīng)用程序通過(guò)在Java代碼中嵌入字符串來(lái)生成HTML。例1說(shuō)明了從servlet生成簡(jiǎn)單的HTML表的過(guò)程。

代碼如下:

MyTableData tableData = MyDAO.queryData();
            PrintWriter writer = response.getWriter();
            writer.println("<table border=\"1\">");
            for (int i = 0; i < tableData.getData().length; i++) {
            writer.println("<tr>");
            writer.println("<td>");
            writer.println(tableData.getData()[i]);
            writer.println("</td>");
            writer.println("</tr>");
            }
            writer.println("</table>");
            

生成的結(jié)果如下所示:

Item 0
Item 1
Item 2
Item 3

例1:一個(gè)從Servlet生成的HTML表

這個(gè)模型存在以下問(wèn)題:

  • 在Java代碼中包含HTML字符串,不僅工作繁重,而且難于閱讀。
  • 每次靜態(tài)字符串改變時(shí),需要重新編譯、打包和部署Java代碼。

許多頁(yè)面包含大量的靜態(tài)的、非生成的文本(需要是可編輯的),以及相對(duì)較少的動(dòng)態(tài)生成的內(nèi)容。為此,在現(xiàn)實(shí)世界中,針對(duì)這個(gè)問(wèn)題的解決方案大都基于頁(yè)面模板的機(jī)制。模板系統(tǒng)使得固定文本和動(dòng)態(tài)動(dòng)作的交叉變得更加自然。在Java世界中,當(dāng)前使用的最受歡迎的模板是JSP。

JSP的早期

借助JSP,HTML和Java代碼可以同時(shí)出現(xiàn)在一個(gè)JSP文件中。JSP文件也稱為JSP頁(yè)面。JSP引擎動(dòng)態(tài)裝載并編譯JSP頁(yè)面。這使得開(kāi)發(fā)流程變得更加簡(jiǎn)易。另外,不要求Java代碼生成靜態(tài)的模板文本。只有動(dòng)態(tài)的動(dòng)作才要求使用Java。

例1可以被重寫(xiě)為下面的JSP 1.1代碼片斷:

<%
            MyTableData tableData = MyDAO.queryData();
            %>
            <table border="1">
            <%
            for (int i = 0; i < tableData.getData().length; i++) {
            %>
            <tr>
            <td>
            <%= tableData.getData()[i] %>
            </td>
            </tr>
            <% }%>
            </table>
            

例2:使用JSP顯示一個(gè)表

HTML代碼被書(shū)寫(xiě)為模板文本。Java代碼被封裝在稱為scirptlet的結(jié)構(gòu)體中。語(yǔ)言的混合晦澀難懂,大部分非Java開(kāi)發(fā)人員對(duì)此都力不能及。這使得在開(kāi)發(fā)過(guò)程中劃分任務(wù)和角色變得困難。

在例2中,業(yè)務(wù)邏輯是直接從JSP頁(yè)面被調(diào)用的。通常,開(kāi)發(fā)人員在JSP頁(yè)面中實(shí)現(xiàn)一些或甚至全部的業(yè)務(wù)邏輯。這并沒(méi)有指定頁(yè)面與業(yè)務(wù)邏輯的交互,因此強(qiáng)制程度降低了很多。對(duì)于大型的項(xiàng)目來(lái)說(shuō),除非開(kāi)發(fā)人員的技術(shù)素養(yǎng)非同一般,否則這種行為將導(dǎo)致代碼變得非模塊化,而且難于維護(hù)。

例2說(shuō)明了以下問(wèn)題必須得到解決:

  • JSP頁(yè)面本身的易讀性。
  • 缺乏分離業(yè)務(wù)邏輯和表示邏輯的模型。

JSP 1.1規(guī)范于1999年11月定稿。自從JSP 1.1以來(lái),JSP最佳實(shí)踐已經(jīng)取得了顯著的進(jìn)展,雖然它們當(dāng)時(shí)實(shí)際上還不存在,但是這種樣式的JSP頁(yè)面仍然普及開(kāi)來(lái)。要解決JSP 1.1的問(wèn)題,必須在JSP技術(shù)上構(gòu)建框架。

Struts和MVC

Struts是實(shí)現(xiàn)了解決這些問(wèn)題的架構(gòu)而且基于JSP的第一批框架之一:

  • 提供有助于執(zhí)行條件、迭代和數(shù)據(jù)選擇的標(biāo)簽庫(kù),增強(qiáng)JSP的易讀性。bean 和logic標(biāo)簽庫(kù)用于完成上述任務(wù)。
  • 使用MVC(模型/視圖/控制器)架構(gòu)。借助MVC,業(yè)務(wù)邏輯和表示邏輯得以分離開(kāi)來(lái)。

在Struts中,MVC架構(gòu)是以頁(yè)面為基礎(chǔ)工作的。它是通過(guò)叫做Model 2的混合架構(gòu)來(lái)實(shí)現(xiàn)的,涉及到servlet與JSP文件的協(xié)作。

下圖說(shuō)明了在Struts中,MVC的不同部分(模型、視圖和控制器)之間的關(guān)系。Struts提供了控制器servlet。模型是在一個(gè)實(shí)現(xiàn)或調(diào)用業(yè)務(wù)邏輯的動(dòng)作類中實(shí)現(xiàn)的,而視圖是在一個(gè)JSP頁(yè)面中實(shí)現(xiàn)的。您仍然能夠在JSP頁(yè)面中實(shí)現(xiàn)業(yè)務(wù)邏輯,但實(shí)際上不鼓勵(lì)這種做法。

圖2:Struts、MVC和Model 2

MVC架構(gòu)相當(dāng)重要,因?yàn)樗鼘⑹勾a自然而然地模塊化,不論是開(kāi)發(fā)小組還是單個(gè)開(kāi)發(fā)人員,都可以從中獲益匪淺。當(dāng)幾個(gè)開(kāi)發(fā)人員一起工作時(shí),較容易劃分角色:一個(gè)開(kāi)發(fā)人員專門(mén)處理業(yè)務(wù)邏輯,同時(shí)其他人負(fù)責(zé)構(gòu)建實(shí)際的Web頁(yè)面。想要了解有關(guān)MVC的更多信息,請(qǐng)參見(jiàn)本文結(jié)尾的參考資料。

例3顯示了使用Struts重寫(xiě)例2中JSP代碼片斷的結(jié)果:

<table border="1">
            <logic:iterate id="data" name="tableData" property="data">
            <tr>
            <td>
            <bean:write name="data"/>
            </td>
            </tr>
            </logic:iterate>
            </table>
            

例3:使用 Struts顯示一個(gè)表

現(xiàn)在,調(diào)用業(yè)務(wù)邏輯的代碼實(shí)現(xiàn)在一個(gè)單獨(dú)的動(dòng)作類(沒(méi)有給出)中,這個(gè)動(dòng)作類傳遞一個(gè)名為tableData的JavaBean給JSP頁(yè)面。例4顯示了上述bean的樣子:

public class TableData {
            private String[] data;
            public String[] getData() {
            return data;
            }
            ...
            }
            

例4:簡(jiǎn)單的 JavaBean

logic:iterate標(biāo)簽的name和property屬性指定了bean的data屬性元素上的一次迭代。property屬性還支持嵌套的和索引的引用,它們構(gòu)成了JSP 2.0中使用的表達(dá)式語(yǔ)言(Expression Language,EL)和JSTL的實(shí)際先驅(qū)。例3說(shuō)明,通過(guò)實(shí)現(xiàn)MVC并提供足夠的標(biāo)簽庫(kù),您可以在JSP文件中完全不使用Java代碼。

步入JSTL和JSP 2.0

Struts標(biāo)簽庫(kù)和表達(dá)式語(yǔ)言存在限制??紤]下面例子中的代碼,它的功能是使用另外一列和交替的行顏色構(gòu)造一個(gè)HTML表:

代碼如下:

<%
            MyTableData tableData = MyDAO.queryData();
            %>
            <table border="1">
            <%
            for (int i = 0; i < tableData.size(); i++) {
            String cellColor = (i % 2 == 0) ? "gray" : "white";
            %>
            <tr>
            <td bgcolor="<%= cellColor %>">
            <%= i %>
            </td>
            <td bgcolor="<%= cellColor %>">
            <%= tableData.get(i) %>
            </td>
            </tr>
            <% }%>
            </table>
            

生成了如下所示的結(jié)果:

0 Item 0
1 Item 1
2 Item 2
3 Item 3

例5:使用Scriptlet的復(fù)雜例子

在例5中,不能使用Struts標(biāo)簽替換所有的scriptlet,因?yàn)榈ㄒ粋€(gè)數(shù)據(jù)元素、一個(gè)位置索引和一次Struts標(biāo)簽庫(kù)不能處理的計(jì)算。

針對(duì)這些限制的解決方案是引入JSTL和JSP 2.0。JSLT實(shí)質(zhì)上是Struts標(biāo)簽庫(kù)的一個(gè)經(jīng)過(guò)深思熟慮的更好的版本。標(biāo)簽的集合更加一致而完備。JSTL引入了表達(dá)式語(yǔ)言(EL),這種語(yǔ)言松散地基于JavaScript和XPath。JSP 2.0天生就支持這種EL。

除了對(duì)EL的天生支持,JSP 2.0還具有以下新特性:

特 性 描 述
表達(dá)式語(yǔ)言API 標(biāo)簽庫(kù)實(shí)現(xiàn)者能夠輕易地利用JSP 2.0的 EL實(shí)現(xiàn)。
實(shí)現(xiàn)定制JSP動(dòng)作的新工具
  • 簡(jiǎn)化的Java API使得在Java中書(shū)寫(xiě)定制標(biāo)簽更加容易。
  • 開(kāi)發(fā)人員能夠在標(biāo)簽文件中書(shū)寫(xiě)定制的JSP動(dòng)作,而不用書(shū)寫(xiě)、編譯和打包Java應(yīng)用程序。
增強(qiáng)的可配置性 在單個(gè)頁(yè)面或在整個(gè)應(yīng)用程序中,顯式地禁止使用scriptlets 很容易做到。

另外還有一些與使用JSTL和EL相關(guān)的優(yōu)點(diǎn):

優(yōu) 點(diǎn) 描 述
XML 和 HTML的一致性 JSTL/EL聯(lián)合起來(lái)可以產(chǎn)生酷似XML文檔的JSP頁(yè)面。結(jié)果不僅更易于閱讀,而且更易于被開(kāi)發(fā)工具處理。例如,圖形工具能夠檢測(cè)或插入迭代器標(biāo)簽,而如果使用Java書(shū)寫(xiě)迭代,迭代器標(biāo)簽便越發(fā)復(fù)雜得多。
Language constraints EL將自身的作用范圍限制在表達(dá)式上。標(biāo)簽庫(kù)提供一個(gè)定義良好的動(dòng)作集合。這很有益處,因?yàn)镴ava誘使Web頁(yè)面作者在頁(yè)面中書(shū)寫(xiě)業(yè)務(wù)邏輯,從而打破了MVC模型。

例5可以使用JSTL標(biāo)簽庫(kù)和JSP 2.0全部重寫(xiě),如例6所示:

<table border="1">
            <c:forEach items="${tableData.data}" var="currentData" varStatus="status">
            <c:set var="cellColor">
            <c:choose>
            <c:when test="${status.index % 2 == 0}">gray</c:when>
            <c:otherwise>white</c:otherwise>
            </c:choose>
            </c:set>
            <tr>
            <td bgcolor="${cellColor}">
            ${status.index}
            </td>
            <td bgcolor="${cellColor}">
            ${currentData}
            </td>
            </tr>
            </c:forEach>
            </table>
            

例6:使用 JSTL和 JSP 2.0書(shū)寫(xiě)的例子

最新的規(guī)范明確支持書(shū)寫(xiě)腳本更少的頁(yè)面,這標(biāo)志著JSP開(kāi)發(fā)的一個(gè)轉(zhuǎn)折點(diǎn)。從純化論者認(rèn)為的HTML"tag soup"與Java scriptlet的無(wú)形混合,到幾乎純粹的命名空間感知的XML文件,JSP頁(yè)面在最佳實(shí)踐方面發(fā)生了重大變化。結(jié)果是語(yǔ)法更加一致,頁(yè)面作者的工作負(fù)擔(dān)減輕,與工具的集成更加容易,以及可以更好地控制業(yè)務(wù)邏輯與表示邏輯的分離。

JSP的前景

對(duì)于與用戶的交互有限而且簡(jiǎn)單的動(dòng)態(tài)Web頁(yè)面來(lái)說(shuō),JSP 2.0、JSTL和MVC就能提供您所需要的全部功能。但是如果要開(kāi)發(fā)復(fù)雜的、基于Web的用戶界面,您不得不做大量的工作,以便提取請(qǐng)求參數(shù)、驗(yàn)證并處理它們、將它們呈現(xiàn)回為HTML控件,等等。整個(gè)過(guò)程冗長(zhǎng)乏味,而且極易出現(xiàn)錯(cuò)誤。

長(zhǎng)期以來(lái),桌面應(yīng)用程序已經(jīng)使用了基于組件的層次MVC框架,比如降低構(gòu)建用戶界面的難度的Swing。此類框架將用戶界面的各個(gè)元素公開(kāi)為以Container/Containee關(guān)系組織的組件。這種方法為自身提供了使用圖形工具構(gòu)建用戶界面的能力。它還允許組件重用,以及在開(kāi)發(fā)過(guò)程中進(jìn)行更好的角色分離。

圖3:包含控件的窗體

通常,Web應(yīng)用程序具有簡(jiǎn)化的控件集合和有限的交互性,但在其他方面與桌面應(yīng)用程序區(qū)別不大。我們很自然地想到,要試著將在桌面上使用的模型應(yīng)用程序到服務(wù)器端的世界中去。這也是JSF專家組選出的解決方案。服務(wù)器端的應(yīng)用程序與桌面應(yīng)用程序相去甚遠(yuǎn),所以JSF需要一個(gè)新的處理模型及API,從而代替Swing中模型和API的使用。

像Struts一樣,JSF符合MVC架構(gòu)。它提供一個(gè)控制器servlet,并且允許使用呈現(xiàn)程序的視圖和使用組件類及基于事件的機(jī)制的模型分離開(kāi)來(lái)。借助JSF,MVC可以工作在單獨(dú)組件的層次上。

開(kāi)發(fā)人員能夠在JSP文件中使用XML標(biāo)記來(lái)描述組件之間的相互關(guān)系,而不用書(shū)寫(xiě)Java代碼來(lái)組裝JSF組件,而后者正是Swing的做法。聯(lián)合使用JSF與JSP是可選的,但這是眾望所歸的首選技術(shù)。例如,JSF附帶有一組標(biāo)準(zhǔn)的用戶界面組件,可以在JSP中通過(guò)Standard JSF Tag Library來(lái)使用它們。這允許不擅長(zhǎng)Java的web頁(yè)面作者也能從事開(kāi)發(fā)復(fù)雜用戶界面的任務(wù)。它還使得實(shí)現(xiàn)圖形工具更加容易。本文稍后將討論這樣的一個(gè)例子。

當(dāng)與JSP一起使用時(shí),因?yàn)橥瑫r(shí)使用了控制器servlet和JSP頁(yè)面,JSF可以被看作是一個(gè)Model 2架構(gòu)。

JSF的承諾包括重用、角色分離和易于使用的工具,從而降低了實(shí)現(xiàn)服務(wù)器端用戶界面的門(mén)檻,并縮短了其所需的時(shí)間。

本文的余下部分專門(mén)講述JSF特殊的呈現(xiàn)方面。想要了解JSF架構(gòu)的更多信息,包括它的API和處理模型,請(qǐng)參見(jiàn)本文結(jié)尾部分的參考資料。

JSF呈現(xiàn)

用戶界面的呈現(xiàn)在于生成信息以使其對(duì)用戶可見(jiàn)。在服務(wù)器端編程領(lǐng)域中,信息通常包括HTML、CSS和JavaScript。呈現(xiàn)涉及到發(fā)送字符流給Web瀏覽器。

呈現(xiàn)JSF層次是一個(gè)遞歸過(guò)程,在此過(guò)程中,每個(gè)使用Java書(shū)寫(xiě)的組件都有機(jī)會(huì)呈現(xiàn)自身。JSF提供了兩個(gè)選擇:

  • 直接從組件呈現(xiàn)。這種做法不能修改組件本身的代碼,所以也就不能修改呈現(xiàn)的樣式。
  • 委派給聚集在稱為RenderKit的實(shí)體中的單獨(dú)呈現(xiàn)程序類。組件的代碼不需要修改,但是改變樣式仍然需要修改RenderKit中的Java代碼。

這兩個(gè)選擇都不是基于模板的,而且就像在剛開(kāi)始使用servlet的時(shí)候,它們還要求非標(biāo)準(zhǔn)的解決方案或者從Java代碼生成HTML。早期訪問(wèn)版本的JSF附帶的呈現(xiàn)程序便是以這種方式實(shí)現(xiàn)的。無(wú)論何時(shí)呈現(xiàn)需要改變,組件開(kāi)發(fā)人員都不得不卷入到其中。與JSP日益提倡的角色分離相比,這是一種嚴(yán)重的,而且?guī)缀跏遣荒芾斫獾牡雇恕?/p>

HTML有一種伴生語(yǔ)言,稱為CSS,特別為樣式而設(shè)計(jì)。假如在JSF的HTML RrnderKit for the Standard User Interface Component中,能夠使用CSS支持,那么CSS便能夠解除JSF呈現(xiàn)的限制。例如,可以使用支持CSS的瀏覽器中的style屬性來(lái)選擇按鈕的背景顏色:

<input type="button" value="Button 1" style="color: red"/>
            <input type="button" value="Button 2" style="color: blue"/>
            

例7:使用 CSS選擇樣式

通過(guò)允許保持組件的呈現(xiàn)代碼不變,即便是在另外的呈現(xiàn)方面改變時(shí)也是如此,CSS的使用部分地解決了樣式問(wèn)題。這種方法具有以下限制:

  • 您不能控制使用CSS的樣式的每個(gè)方面。
  • 并非每個(gè)被部署的Web瀏覽器都支持所有的CSS規(guī)范。

為了說(shuō)明第一點(diǎn),考慮一下,您可能想要應(yīng)用程序中的每個(gè)表都交替顯示行的顏色,就像例5和6那樣。CSS不支持描述這種行為。

對(duì)于不支持最新版本CSS的Web瀏覽器來(lái)說(shuō),一個(gè)流行的不使用CSS的HTML技巧便是通過(guò)嵌入表來(lái)創(chuàng)建表邊界。

下面的代碼:

<table cellpadding="0" cellspacing="0" border="0" bgcolor="red">
            <tr>
            <td>
            <!-- Original table -->
            <table width="100%">
            <tr>
            <td bgcolor="white">1.1</td>
            <td bgcolor="white">1.2</td>
            </tr>
            <tr>
            <td bgcolor="white">2.1</td>
            <td bgcolor="white">2.2</td>
            </tr>
            </table>
            </td>
            </tr>
            </table>
            

生成的結(jié)果如下所示:

1.1 1.2
2.1 2.2

例8:嵌入表

要處理這種使用JSTL的情況,代碼不得不在應(yīng)用程序的每個(gè)JSP頁(yè)面中被重復(fù)。如果在開(kāi)發(fā)過(guò)程中,開(kāi)發(fā)人員不得不在不同的樣式之間來(lái)回切換好幾次,例如要處理變化的需求時(shí),修改必須在整個(gè)應(yīng)用程序中重復(fù)進(jìn)行。借助JSF,可以構(gòu)建一個(gè)定制的組件,但是這意味著,您必須書(shū)寫(xiě)復(fù)雜的Java代碼。如果要實(shí)現(xiàn)這種類型的樣式技術(shù),兩種技術(shù)都幫不上什么忙。

至少有兩種解決方案是可用的:

  • 在類似于JSP 2.0標(biāo)簽文件的模板文件中實(shí)現(xiàn)JSF呈現(xiàn)程序。不論是JSF還是JSP,都不支持這樣的模型,而且在JSF 1.0的最終版本發(fā)行之前,JSF呈現(xiàn)不可能有顯著的改變。
  • 使用XML處理。這種解決方案又稱為Model 2X,將在下一節(jié)中討論。

Model 2X

架構(gòu)

Model 2X最重要的方面是XML轉(zhuǎn)換語(yǔ)言的使用,比如XSL轉(zhuǎn)換(XSL Transformation,XSLT),連同servlet和JSP。下圖說(shuō)明了Model 2X的基本架構(gòu):

圖4:Model 2X架構(gòu)

servlet或JSP的輸出通常被直接發(fā)送給Web瀏覽器。有了Model 2X之后,在到達(dá)Web瀏覽器之前,輸出先要經(jīng)由一個(gè)XML轉(zhuǎn)換的階段。為了理解這一點(diǎn),考慮JSP和XML兩個(gè)世界之間的關(guān)系相當(dāng)重要。

JSP和XML

設(shè)計(jì)JSP的目的是為了能夠產(chǎn)生任意的文本格式,比如HTML、逗號(hào)分隔值(Comma-Separated Value,CSV)以及層疊樣式表(Cascading Style Sheet,CSS)。出于這個(gè)理由,您可以使用JSP生成XML文檔。要注意的地方是,JSP不能強(qiáng)制生成格式良好的XML輸出。為了生成格式良好的XML,您不得不采取以下預(yù)防措施:

  • JSP中直接呈現(xiàn)的任何模板文本標(biāo)記必須是格式良好的。
  • 標(biāo)簽庫(kù)必須生成格式良好的XML。

考慮下面這個(gè)例子,即包含有效HTML標(biāo)簽的JSP代碼片斷:

<body>
            <hr>
            </body>
            

有一種方法可以確保模板文本是格式良好的XML,即書(shū)寫(xiě)Sun所號(hào)稱的JSP文檔,而不是常規(guī)的JSP頁(yè)面。JSP文檔不過(guò)就是JSP頁(yè)面的XML版本而已。JSP結(jié)構(gòu)體,比如<% %><%@ %>,不符合XML規(guī)范。在JSP文檔中,它們被相應(yīng)的XML標(biāo)簽所代替,比如<jsp:directive.include><jsp:scriptlet>標(biāo)簽。書(shū)寫(xiě)JSP文檔并不是從JSP生成格式良好的XML的要求,但是因?yàn)镴SP引擎要確保文檔符合XML規(guī)范,它在前面顯示了許多錯(cuò)誤。

標(biāo)簽庫(kù)經(jīng)常生成HTML。Struts的html標(biāo)簽庫(kù)和標(biāo)準(zhǔn)JSF標(biāo)簽庫(kù)便是如此。使用這種標(biāo)簽庫(kù)時(shí),必須修改它們,以便生成XML,而不是HTML。借助JSF可以輕松完成這項(xiàng)工作,具體方法是書(shū)寫(xiě)一個(gè)XML呈現(xiàn)程序,并用它來(lái)代替現(xiàn)有的HTML呈現(xiàn)程序。本文稍后將詳細(xì)討論這個(gè)問(wèn)題。

執(zhí)行JSP頁(yè)面的結(jié)果仍然會(huì)得到字符流,即便該頁(yè)面符合XML規(guī)范也是如此。不能確保輸出是有效的,或者甚至是格式良好的XML。如果工具需要將JSP輸出處理為XML,如Model 2X中所示,這時(shí)便需要一個(gè)解析的步驟。

XSLT

XML被JSP頁(yè)面生成之后,它就可以被其他工具消費(fèi)。XSLT便是一個(gè)這樣的工具,一種被設(shè)計(jì)用來(lái)執(zhí)行XML文檔轉(zhuǎn)換的功能語(yǔ)言。

一個(gè)簡(jiǎn)單的XSLT轉(zhuǎn)換可以通過(guò)下面這種方式來(lái)表示:

圖5:簡(jiǎn)單的XSLT轉(zhuǎn)換

XSLT轉(zhuǎn)換的結(jié)果可能是XML、HTML或者其他任意的文本格式。

XSLT的1.0版本發(fā)布于1999年11月。從那之后,為了將它發(fā)展為XSLT 2.0規(guī)范,許多人付出了大量的勞動(dòng)。XSLT 2.0基于XPath 2.0,是一種新版本的XML表達(dá)式語(yǔ)言,具有更好的一致性,而且包含1.0版本中所沒(méi)有的一些特性,比如分組和用戶定義的XPath函數(shù)。

一個(gè)XSLT程序被稱為一個(gè)樣式表。這個(gè)術(shù)語(yǔ)在XSLT 2.0規(guī)范草案中有著清晰的表述:

"術(shù)語(yǔ)樣式表反映了這樣一個(gè)事實(shí),即XSLT的重要角色之一便是為XML源文檔添加樣式信息,具體方法是將它轉(zhuǎn)換為由XSL格式化對(duì)象(參見(jiàn)[XSL格式化對(duì)象])組成的文檔,或者轉(zhuǎn)換為另一種面向表示的格式,比如HTML、XHTML或SVG。"

因此,XSLT這種語(yǔ)言的設(shè)計(jì)目的相當(dāng)明確,就是為了解決上述的樣式問(wèn)題。

JSF和Model 2X

現(xiàn)在,您已經(jīng)能夠理解在Model 2X中,XML和XSLT技術(shù)是如何與JSP和JSF進(jìn)行交互的。下面的JSP代碼片斷使用了JSF,這段代碼來(lái)自當(dāng)前JSF引用實(shí)現(xiàn)中的"guessNumber"例子。它包含了HTML模板代碼和構(gòu)建簡(jiǎn)單窗體的JSF標(biāo)簽:

<%@ taglib uri="http://java./j2ee/html_basic/" prefix="faces"%>
            <html>
            <head><title>Hello</title></head>
            <body>
            <h1>Hello</h1>
            <faces:usefaces>
            <faces:form id="helloForm" formName="helloForm">
            <table>
            <tr>
            <td>
            <faces:textentry_input id="userNo"
            modelReference="UserNumberBean.userNumber"/>
            </td>
            <td>
            <faces:command_button id="submit" commandName="submit"/><p>
            </td>
            </tr>
            </table>
            <faces:validation_message componentId="userNo"/>
            </faces:form>
            </faces:usefaces>
            </body>
            </html>
            

例9:JSF 例子

執(zhí)行使用XML呈現(xiàn)程序的代碼片斷,可能會(huì)輸出以下格式良好的XML文檔:

<html>
            <head><title>Hello</title></head>
            <body>
            <h1>Hello</h1>
            <form method="post" action="/guessNumber/faces/greeting.jsp">
            <table>
            <tr>
            <td>
            <input type="text" name="/helloForm/userNo"/>
            </td>
            <td>
            <input type="submit" name="submit" value="submit"/>
            </td>
            </tr>
            </table>
            </form>
            </body>
            </html>
            

例10:JSF例子的輸出

當(dāng)JSP輸出被解析并被反饋到XSLT轉(zhuǎn)換器時(shí),您可以輕松地應(yīng)用程序樣式。下圖說(shuō)明了處理模型,同時(shí)也是Model 2X的一個(gè)特殊例子:

圖6:JSF和XSLT呈現(xiàn)模型

例如,要將應(yīng)用程序中的每個(gè)按鈕呈現(xiàn)為紅色,使用下面的XSLT模板即可達(dá)到目的:

<!-- Find all input elements with a "submit" type attribute -->
            <xsl:template match="input[@type = ‘submit‘]">
            <!-- Copy the element found -->
            <input>
            <!-- Copy all existing attributes -->
            <xsl:copy-of select="@*"/>
            <!-- Add a "style" attribute -->
            <xsl:attribute name="style">color: red</xsl:attribute>
            </input>
            </xsl:template>
            

例10:使用一個(gè)XSLT 模板添加樣式

XSLT模板并不限于在由JSF呈現(xiàn)程序生成的標(biāo)記上使用。例如,它可以使用同樣的技術(shù)生成頁(yè)面布局。例11說(shuō)明了如何插入一個(gè)表,該表包含一個(gè)標(biāo)題行以及包含原始文檔主體的第二行:

<!-- Match the body element -->
            <xsl:template match="body">
            <!-- Copy the element found and set a background color -->
            <body bgcolor="white">
            <table border="0">
            <!-- Create a header row -->
            <tr>
            <td>Header</td>
            </tr>
            <tr>
            <td>
            <!-- Copy the actual body -->
            <xsl:apply-templates/>
            </td>
            </tr>
            </table>
            </body>
            </xsl:template>
            

例11:創(chuàng)建一個(gè)頁(yè)面布局

Model 2X的優(yōu)點(diǎn)包括:

優(yōu) 點(diǎn) 描 述
更好的模塊化和靈活性 樣式可以被集中化。書(shū)寫(xiě)一個(gè)樣式表就可以處理一個(gè)應(yīng)用程序的所有外觀。外觀可以在整個(gè)應(yīng)用程序中被重用。運(yùn)行時(shí),同一應(yīng)用程序中的不同外觀可以按照需要進(jìn)行切換。
快速轉(zhuǎn)向 無(wú)須對(duì)Java代碼做任何修改。JSF 呈現(xiàn)程序只要書(shū)寫(xiě)一次,就可以被所有頁(yè)面使用。當(dāng)應(yīng)用程序正在運(yùn)行時(shí),XSLT樣式表可以被重新裝載,而且變化幾乎是瞬時(shí)的。
強(qiáng)大的表現(xiàn)功能 XSLT是一種特別適合于操作XML的語(yǔ)言,也因此適合操作XHTML。操作的可能性是無(wú)窮的。XPath是一種表達(dá)式語(yǔ)言,它比JSP 2.0的EL以及JSTL要更加先進(jìn)。
標(biāo)準(zhǔn)的一致性 XSLT是一個(gè)W3C標(biāo)準(zhǔn),被廣泛采用。這意味著多個(gè)廠商針對(duì)它開(kāi)發(fā)出了大量可用的文檔和工具,包括幾種針對(duì)Java的優(yōu)秀的XSLT轉(zhuǎn)換工具。Java API for XML Processing (JAXP)支持XSLT轉(zhuǎn)換。Xalan是一款流行的XSLT轉(zhuǎn)換工具,它與J2SE 1.4捆綁在一起,而且所有一流的應(yīng)用程序服務(wù)器都支持XSLT。
管道技術(shù) XSLT 適合以管道的方式進(jìn)行處理。下面將詳細(xì)討論XML管道。
Familiarity 在特定的條件和迭代器中,幾個(gè)JSTL結(jié)構(gòu)體看起來(lái)很像XSLT結(jié)構(gòu)體。這使得熟悉JSTL的人學(xué)習(xí)XSLT會(huì)更加輕松,反之亦然。

Model 2X的以下限制應(yīng)該引起注意:

限 制 描 述
輔助處理 Model 2X在表示邏輯的執(zhí)行過(guò)程中引入了輔助步驟。大多數(shù)時(shí)候,性能提升十分有限,而靈活的貿(mào)易性能才是主要的方面。
學(xué)習(xí)曲線 必須學(xué)習(xí)XSLT。對(duì)于大多數(shù)用途來(lái)說(shuō),學(xué)習(xí)XSLT是如此容易,以至于這根本就不成為問(wèn)題。
要求 因?yàn)镴SP并不強(qiáng)制輸出格式良好的XML,所以這個(gè)任務(wù)便落到了開(kāi)發(fā)人員的肩上。

XML 管道

Model 2X的使用為JSP處理模型引入了一個(gè)新的維度,因?yàn)樗婕暗綆讉€(gè)順序執(zhí)行的步驟:

  1. 處理整個(gè)JSP頁(yè)面。根據(jù)常規(guī)的JSP處理模型,模板文本和標(biāo)簽庫(kù)產(chǎn)生輸出。
  2. 輸出被解析為XML。
  3. 執(zhí)行XML處理。

實(shí)際上構(gòu)建的是一個(gè)兩段式管道??梢詾槊總€(gè)階段分配定義良好的角色:

管道的概念使Web應(yīng)用程序的表示層變得模塊化。不用修改JSP頁(yè)面,管道就可以被擴(kuò)展為包含具有不同角色的另外階段,例如:

角 色 描 述
本地化 在單獨(dú)的管道階段中執(zhí)行本地化允許本地化JSP標(biāo)簽庫(kù)的輸出,以及修改頁(yè)面布局的高級(jí)本地化。阿拉伯語(yǔ)的Web頁(yè)面可能在右側(cè)呈現(xiàn)其導(dǎo)航欄,而不是左側(cè)。
多種瀏覽器支持 對(duì)于支持諸如最新版本的CSS這樣的高級(jí)呈現(xiàn)功能的Web瀏覽器來(lái)說(shuō),一個(gè)樣式表可以為其生成輸出,而另一個(gè)樣式表可以為老式瀏覽器生成輸出。
多種設(shè)備支持 一個(gè)單獨(dú)的管道能夠以完全不同的設(shè)備作為目標(biāo),比如PDA或手機(jī)。
包含 在一個(gè)管道階段中,可以從其他JSP頁(yè)面構(gòu)建出一個(gè)頁(yè)面,包括頭、腳和主頁(yè)主體。

管道的一個(gè)完整例子如圖7所示:

圖7:管道的例子

實(shí)現(xiàn)Model 2X

使用JSF實(shí)現(xiàn)Model 2X要求:

  1. 為JSF Standard User Interface Component創(chuàng)建一個(gè)XML RenderKit。
  2. 連接一個(gè)XML解析器,以及一個(gè)XSLT轉(zhuǎn)換器(簡(jiǎn)單實(shí)現(xiàn))或XML管道(高級(jí)實(shí)現(xiàn))。

下面的代碼說(shuō)明了對(duì)第一步的需要。JSF faces:textentry_input標(biāo)簽不能生成格式良好的XML輸出:

<INPUT TYPE="text" NAME="name" VALUE="value">
            

生成XML的JSF RenderKit必須生成格式良好的XML元素和屬性。另外,它應(yīng)該遵照以小寫(xiě)形式創(chuàng)建元素的XHTML慣例:

<input type="text" name="name" value="value"/>
            

JSF的當(dāng)前引用實(shí)現(xiàn)沒(méi)有附帶一個(gè)XML RenderKit,但是對(duì)于熟悉JSF的開(kāi)發(fā)人員來(lái)說(shuō),書(shū)寫(xiě)這樣一個(gè)呈現(xiàn)程序不是什么難事。本文中包含了一個(gè)簡(jiǎn)單的RenderKit。

使用以下方法可以完成與XML解析器和XSLT轉(zhuǎn)換器的連接:

  • Servlet過(guò)濾器。除了生成格式良好的XML之外,JSP頁(yè)面不需要?jiǎng)e的任何特殊關(guān)照。標(biāo)簽庫(kù)。一個(gè)封裝了整個(gè)頁(yè)面的標(biāo)簽必須被插入到每一個(gè)JSP頁(yè)面中。您可以使用JSP 2.0的抬頭與結(jié)尾機(jī)制來(lái)避免代碼重復(fù)。
  • 您可以基于DOM或SAX來(lái)構(gòu)建XML管道邏輯。您可以創(chuàng)建定制的代碼,或者使用現(xiàn)有的解決方案,比如OXF或Cocoon。

想要了解更多細(xì)節(jié),請(qǐng)參見(jiàn)本文中使用的示例源代碼。

結(jié)束語(yǔ)

這些年來(lái),JSP技術(shù)已經(jīng)趨于成熟。最佳實(shí)踐已經(jīng)表現(xiàn)為在真實(shí)應(yīng)用程序中使用JSP的結(jié)果,這導(dǎo)致JSP技術(shù)的核心得到了很大的增強(qiáng),并使諸如JSTL這樣的新規(guī)范得到發(fā)展。JSF引入了一個(gè)期待已久的用戶界面組件模型,但是JSF也表現(xiàn)出其在呈現(xiàn)架構(gòu)中受到限制的地方。眾多開(kāi)發(fā)人員可能認(rèn)為這些限制是一個(gè)概念上的倒退。

Java開(kāi)發(fā)人員相當(dāng)熟悉XML技術(shù),但他們沒(méi)有必要了解XML是如何被用于增強(qiáng)Web應(yīng)用程序的表示層的。Model 2X是一個(gè)簡(jiǎn)單的架構(gòu),它構(gòu)建在JSP、JSF和XML技術(shù)的基礎(chǔ)之上,減少了JSF呈現(xiàn)模型的限制,并為J2EE應(yīng)用程序提供了一個(gè)靈活的表示層。

盡管JSP有逐漸與XML同化的趨勢(shì),JSP 2.0還是沒(méi)有與XML標(biāo)準(zhǔn)完全集成。將來(lái),通過(guò)將Model 2X添加到JSP引擎的核心中,JSP應(yīng)該完成這種演變。

參考資料

JavaServer Pages (JSP)
http://java./products/jsp/

Servlet 技術(shù)
http://java./products/servlet/

JSTL 規(guī)范
http:///aboutJava/communityprocess/final/jsr052/

JavaServer Faces (JSF)
http://java./j2ee/javaserverfaces/

Struts和XSLT方面的JavaWorld文章
http://www./javaworld/jw-02-2002/jw-0201-strutsxslt.html

初識(shí) JavaServer Faces(第1部分)
http://www./javaworld/jw-11-2002/jw-1129-jsf.html

XSLT 1.0和XSLT 2.0 (草案)
http://www./TR/xslt, http://www./TR/xslt20/

OXF
http://www./oxf/

Model 2X
http://www./model2x

使用XML管道構(gòu)建Pet Store
http://www./content/orbeon/pet_store.jsp

XML管道簡(jiǎn)介
http://www./oxf/whitepaper

Struts
http://jakarta./struts/

The Model-View-Controller (‘MVC‘) 設(shè)計(jì)模式
http://jakarta./struts/userGuide/introduction.html#mvc

Apache Xalan for Java
http://xml./xalan-j/index.html

Saxon - The XSLT 處理器
http://saxon./

Cocoon
http://xml./cocoon/

關(guān)于作者

Erik Bruchez專門(mén)研究Web應(yīng)用程序架構(gòu)。他是OXF的聯(lián)合架構(gòu)師。OXF是由Orbeon, Inc開(kāi)發(fā)的一個(gè)基于XML的應(yīng)用程序框架。Orbeon, Inc是一家J2EE和XML咨詢公司,創(chuàng)立于1999年,Erik正是這家公司的聯(lián)合創(chuàng)始人之一。Erik曾領(lǐng)導(dǎo)過(guò)幾個(gè)軟件平臺(tái)和應(yīng)用程序的設(shè)計(jì)與實(shí)現(xiàn),其中包括視頻機(jī)頂盒(set-top box)和供應(yīng)鏈管理應(yīng)用程序方面的中間件。在Orbeon之前,Erik曾為Symantec Corp.工作,并為VisualCafe產(chǎn)品線做出了貢獻(xiàn)。Erik從位于瑞士洛桑的瑞士技術(shù)學(xué)院(EPFL)獲得了計(jì)算機(jī)科學(xué)碩士學(xué)位。

Omar Tazi曾在位于Silicon Valley心臟地帶的幾家高科技公司中擔(dān)任過(guò)高級(jí)管理職位,包括Symantec、Oracle和WebGain。Omar是Java Community Process Executive Committee的成員,還是眾多JSRs(Java Specification Requests)的活躍成員,并且成為了其中Expert Group的成員。他還經(jīng)常在BEA eWorld和JavaOne這樣的軟件年會(huì)上發(fā)言。他的主要興趣在于Java 2 Platform Enterprise Edition(J2EE)、XML和Web service等方面。在橫跨大西洋之前,Omar曾在位于洛桑的瑞士聯(lián)邦技術(shù)學(xué)院中擔(dān)任幾個(gè)研究和教學(xué)職位,專攻人工智能算法和面向?qū)ο蟮木幊獭K麚碛杏?jì)算機(jī)科學(xué)和電子工程的碩士學(xué)位。

    本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購(gòu)買(mǎi)等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
    轉(zhuǎn)藏 分享 獻(xiàn)花(0

    0條評(píng)論

    發(fā)表

    請(qǐng)遵守用戶 評(píng)論公約

    類似文章 更多