簡(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é)果如下所示:
例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í)行的步驟:
- 處理整個(gè)JSP頁(yè)面。根據(jù)常規(guī)的JSP處理模型,模板文本和標(biāo)簽庫(kù)產(chǎn)生輸出。
- 輸出被解析為XML。
- 執(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要求:
- 為JSF Standard User Interface Component創(chuàng)建一個(gè)XML RenderKit。
- 連接一個(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é)位。