Java在最近幾年逐漸升溫,隨著Java SE 5和Java SE 6的推出,Java的未來更顯得無比輝煌。但以Java為基礎的JSP在Java SE 5推出之前卻一直抬不起頭來,這最重要的原因就是JSP雖然功能十分強大,但大的優(yōu)點也是它的大缺點,功能強大就意味著復雜,尤其是設計前端界面的可視化工具不多,也不夠強大。因此,設計JSP頁面就變得十分復雜和繁瑣。不過,在Java SE 5推出的同時,Sun為了簡化JSP的開發(fā)難度,推出了新的JavaServer Faces(簡稱JSF)規(guī)范。從而使JSP走上了康莊大道。
一、什么是JSF
JSF和JSP是一對新的搭檔。JSP是用于后臺的邏輯處理的技術,而JSF恰恰相反,是使開發(fā)人員能夠快速的開發(fā)基于 Java 的 Web 應用程序的技術,是一種表現(xiàn)層技術。目前,JSF1.2已經(jīng)正式作為一個標準加入了Java EE 5中。
作為一種高度組件化的技術,開發(fā)人員可以在一些開發(fā)工具的支持下,實現(xiàn)拖拉式編輯操作,用戶只需要簡單的將 JSF 組件拖到頁面上,就可以很容易的進行 Web 開發(fā)了。這是其作為一種組件化的技術所具有的大好處,我們能用的組件不光是一些比較簡單的輸入框之類,還有更多復雜的組件可以使用的,比如 DataTable 這樣的表格組件, Tree 這樣的樹形組件等等。
作為一種標準的技術,JSF還得到了相當多工具提供商的支持。同時我們也會有很多很好的免費開發(fā)工具可以使用,前不久 Sun Java Studio Creator 2 和 Oracle JDeveloper 10g 作為免費的支持 JSF 的開發(fā)工具發(fā)布,給 JSF 帶來了不小的生氣。另外我們也有一些很優(yōu)秀的商業(yè)開發(fā)工具可共選擇,BEA Workshop (原 M7 NitroX),Exadel,MyEclipse 這樣的基于 Eclipse 的插件開發(fā)工具,為現(xiàn)在廣大的 Eclipse 用戶帶來了不小的便利,IBM 的 Rational Application Developer 和 Borland 的 JBuilder 也是很不錯的支持 JSF 可視化開發(fā)的商業(yè)開發(fā)工具。
JSF和傳統(tǒng)的Web技術有著本質上的差別,在傳統(tǒng)的Web技術需要用戶自己對瀏覽器請求進行捕捉,保存客戶端狀態(tài),并且手工控制著頁面的轉向,等等。而JSF的出現(xiàn),無疑給我們帶來了巨大的便利,JSF 提供了事件驅動的頁面導航模型,該模型使應用程序開發(fā)人員能夠設計應用程序的頁面流。與 Struts 的方式向類似的是,所有的頁面流信息都定義在 JSF 配置 XML 文件 (faces-config.xml) 中,而非硬編碼在應用程序中。這很大程度簡化了開發(fā)人員開發(fā)難度,簡化了應用程序的開發(fā)。
同時JSF也是一種遵循模型-視圖-控制器 (MVC) 模式的框架。實現(xiàn)了視圖代碼(View)與應用邏輯(Model)的完全分離,使得使用 JSF 技術的應用程序能夠很好的實現(xiàn)頁面與代碼的分離。所有對 JSF 頁面的請求都會通過一個前端控制器 (FacesServlet) 處理,系統(tǒng)自動處理用戶的請求,并將結果返回給用戶。這和傳統(tǒng)的 MVC 框架并沒有太大的區(qū)別。
在JSF中不僅使用了 POJO 技術,而且還使用了類似 Spring 的控制反轉(IoC) (或稱為依賴注入-DI) 技術,在 JSF 的 Backing Bean 中,我們可以把視圖所需要的數(shù)據(jù)和操作放進一個 Backing Bean 中。同時得益于 JSF 使用的 DI 技術,我們可以在配置文件中初始化 Managed Bean,同時我們也可以通過這樣的技術很方便的和使用類似技術的 Spring 進行整合。
二、如何在JSP中使用JSF
JSF只有通過和JSP相結合,才能充分發(fā)揮它的功效。JSF是通過標簽庫和JSP進行集成的。標簽庫就相當于ASP.NET的服務端組件。JSF提供了非常豐富的標簽庫,通過這些標簽庫,可以生成各種客戶端模型,如HTML、WML、XML以及JavaScript等。通過這些標簽,你可以很容易建立大規(guī)模的客戶端模型,并由這些標簽自動處理客戶端請求。
接下來讓我們來看一個如何使JSF和JSP在一起工作的例子。在JSF中有兩個庫。第一個叫做內(nèi)核庫,在這個庫中包含了各種主要的標簽,如配置組件、管理事件、驗證輸入信息等。第二個庫的主要功能是將HTML和JSF的各種標簽相對應。每一個JSF標簽都會對應一個HTML組件。如UIInput標簽對應了HTML中的文本框或密碼框。
在JSF標簽中文本輸入框叫做inputText,而密碼輸入庫叫inputSecret。下面是一個簡單的JSF和JSP結合的用戶接口程序。
<%@ taglib uri="/tupian/20230522/" prefix="h" %> <%@ taglib uri="/tupian/20230522/" prefix="f" %> <html> <head> <meta HTTP-EQUIV="Content-Type" CONTENT="text/html;CHARSET=GB2312"> <title>第一個JSF程序</title> </head> <body> ?。糵:view> <h:form id="simpleForm"> ?。糷:outputText id="favoriteLabel" value="請輸入一個數(shù)字:"/> <h:inputText id="favoriteValue" value="#{simple.longValue}"> ?。糵:validateLongrange maximum="30" minimum="0"/> </h:inputText> ?。紁/> ?。糷:commandButton id="submit" value="提交" action="#{simple.simpleActionMethod}"/> ?。?h:form> ?。?f:view> </body> </html> |
在上面的代碼中,我們可以了解到JSF是如何同JSP集成的。我們首先可以看到一個內(nèi)核標簽:view。然后是幾個JSF組件。如form、outputText、inputText以及commandButton。這幾個組件被放到form中從而開成了form中的一部分。在程序的最開始,必須使用import導入兩個標簽庫。代碼如下。
<%@ taglib uri="/tupian/20230522/" prefix="h" %> <%@ taglib uri="/tupian/20230522/" prefix="f" %> |
上面2行代碼聲明了JSP中要使用哪一個JSF標簽庫。內(nèi)核庫使用前綴(prefix)f聲明,而HTML庫使用前綴(prefix)h聲明。這兩個前綴并不是必須要使用,而只是一個建議。在程序中,內(nèi)核庫必須要使用,因為view在所有的JSF頁中必須使用。而HTML標簽在運行時將JSF標簽轉化為HTML組件,這個h前綴并不是必須的,而是JSF規(guī)范推薦使用的,這樣,我們使我們的JSF程序更易讀。
在聲明后是幾行標準的HTML語句,本文不再詳述。從<f:view>開始,是一段JSF語句。這段代碼如下所示:
<f:view> <h:form id="simpleForm"> <h:outputText id="favoriteLabel" value="請輸入一個數(shù)字:"/> <h:inputText id="favoriteValue" value="#{simple.longValue}"> <f:validateLongrange maximum="30" minimum="0"/> </h:inputText> <p/> <h:commandButton id="submit" value="提交" action="#{simple.simpleActionMethod}"/> </h:form> </f:view> |
?。?f:view>標簽預示著JSF的開始,而它的下一個標簽form將建立一個HTML Form。而outputText標簽相當于HTML中的label組件。inputText標簽相當于HTML中的textField組件。而commandButton標簽相當于HTML中的submit按鈕。
三、JSP如何響應JSF的請求
從上面的例子我們已經(jīng)知道如何在JSP中使用JSF了,在這一部分讓我們來看看在JSF是如何處理請求的。
首先讓我們來看一個例子,這個例子是將華氏度轉換為攝氏度。當用戶點擊提交按鈕時程序將進行轉換。
<%@ taglib uri="/tupian/20230522/" prefix="h" %> <%@ taglib uri="/tupian/20230522/" prefix="f" %> <html> <head> ?。糾eta HTTP-EQUIV="Content-Type" CONTENT="text/html;CHARSET=GB2312"> <title>溫度轉換程序</title> </head> <body> <f:view> ?。糷:form> ?。糳iv> <h:outputText id="fahrenheitLabel" value="請輸入華氏溫度:"/> ?。約pan> <h:inputText id="temperature" value="#{tc.fahrenheitTemp}"> ?。糵:validateDoublerange minimum="-100.0" maximum="100.0"/> ?。糵:valuechangeListener type="tempconv.page.TCChangedListener"/> </h:inputText> ?。?span> </div> ?。糳iv> <h:outputText id="celsiusLabel" value="攝氏溫度:"/> <span> ?。糷:outputText id="celsiusValue" value="#{tc.celsiusTemp}"> ?。糵:convertNumber maxFractionDigits="3" type="number"/> ?。?h:outputText> ?。?span> ?。?div> ?。糳iv> ?。糷:commandButton value="轉換" action="#{tc.convert}"> </h:commandButton> ?。?div> </h:form> ?。?f:view> </body> </html> |
<h:inputText id="temperature" value="#{tc.fahrenheitTemp}"> <f:validateDoublerange minimum="-100.0" maximum="100.0"/> ?。糵:valuechangeListener type="tempconv.page.TCChangedListener"/> </h:inputText> |
<f:valuechangeListener type="tempconv.page.TCChangedListener"/> |
<div> <h:commandButton value="轉換" action="#{tc.convert}"> ?。?h:commandButton> </div> |
public class TCChangedListener implements ValueChangeListener { public TCChangedListener() { super(); } // 事件處理 public void processValueChange(ValueChangeEvent event) throws AbortProcessingException { UIComponent comp = event.getComponent(); Object value = event.getNewValue(); if (null != value) { float curVal = ((Number) value).floatValue(); Map values = comp.getAttributes(); if (curVal < 0) { values.put("styleClass", "red"); } else { values.put("styleClass", "black"); } } } |
名稱欄目:JSP和JSF雙劍合并打造完美Web應用-創(chuàng)新互聯(lián) 文章起源:http://weahome.cn/article/shihe.html 其他資訊 |