表現層框架之爭 JSF與Struts的異同
Struts和JSF/Tapestry都屬于表現層框架,這兩種分屬不同性質的框架,后者是一種事件 驅動 型的組件模型,而Struts只是單純的MVC模式框架,老外總是急吼吼說事件驅動型就比MVC模式框架好,何以見得,我們下面進行詳細分析比較一下到底是怎么回事? 首先事件是
Struts和JSF/Tapestry都屬于表現層框架,這兩種分屬不同性質的框架,后者是一種事件
驅動型的組件模型,而Struts只是單純的MVC模式框架,老外總是急吼吼說事件驅動型就比MVC模式框架好,何以見得,我們下面進行詳細分析比較一下到底是怎么回事?
首先事件是指從客戶端頁面(瀏覽器)由用戶操作觸發的事件,Struts使用Action來接受瀏覽器表單提交的事件,這里使用了Command模式,每個繼承Action的子類都必須實現一個方法execute。
在struts中,實際是一個表單Form對應一個Action類(或DispatchAction),換一句話說:在Struts中實際是一個表單只能對應一個事件,struts這種事件方式稱為application event,application event和component event相比是一種粗粒度的事件。
struts重要的表單對象ActionForm是一種對象,它代表了一種應用,這個對象中至少包含幾個字段,這些字段是Jsp頁面表單中的input字段,因為一個表單對應一個事件,所以,當我們需要將事件粒度細化到表單中這些字段時,也就是說,一個字段對應一個事件時,單純使用Struts就不太可能,當然通過結合
JavaScript也是可以轉彎實現的。
而這種情況使用JSF就可以方便實現,
<h:inputText id="userId" value="#{login.userId}"> ?。糵:valueChangeListener type="logindemo.UserLoginChanged" /> </h:inputText> |
#{login.userId}表示從名為login的JavaBean的getUserId獲得的結果,這個功能使用struts也可以實現,name="login" property="userId"
關鍵是第二行,這里表示如果userId的值改變并且確定提交后,將觸發調用類UserLoginChanged的processValueChanged(...)方法。
JSF可以為組件提供兩種事件:Value Changed和 Action. 前者我們已經在上節見識過用處,后者就相當于struts中表單提交Action機制,它的JSF寫法如下:
<h:commandButton id="login" commandName="login"> ?。糵:actionListener type=”logindemo.LoginActionListener” /> </h:commandButton> |
從代碼可以看出,這兩種事件是通過Listerner這樣觀察者模式貼在具體組件字段上的,而Struts此類事件是原始的一種表單提交Submit觸發機制。如果說前者比較語言化(
編程語言習慣做法類似Swing
編程);后者是屬于WEB化,因為它是來自Html表單,如果你起步是從Perl/
PHP開始,反而容易接受Struts這種風格。
基本配置 Struts和JSF都是一種框架,JSF必須需要兩種包JSF核心包、JSTL包(標簽庫),此外,JSF還將使用到Apache項目的一些commons包,這些Apache包只要部署在你的
服務器中既可。
JSF包
下載地址:http://
java.sun.com/j2ee/
javaserverfaces/download.html選擇其中Reference Implementation。
JSTL包
下載在http://jakarta.apache.org/site/downloads/downloads_taglibs-standard.cgi
所以,從JSF的驅動包組成看,其
開源基因也占據很大的比重,JSF是一個SUN伙伴們工業標準和
開源之間的一個混血兒。
上述兩個地址下載的jar合并在一起就是JSF所需要的全部驅動包了。與Struts的驅動包一樣,這些驅動包必須位于
Web項目的WEB-INF/lib,和Struts一樣的是也必須在
web.xml中有如下配置:
<web-app> ?。約ervlet> ?。約ervlet-name>Faces Servlet</servlet-name> ?。約ervlet-class>javax.faces.webapp.FacesServlet</servlet-class> ?。糽oad-on-startup>1</load-on-startup> ?。?servlet> ?。約ervlet-mapping> ?。約ervlet-name>Faces Servlet</servlet-name> ?。紆rl-pattern>*.faces</url-pattern> ?。?servlet-mapping> </web-app> |
這里和Struts的web.xml配置何其相似,簡直一模一樣。
正如Struts的struts-config.xml一樣,JSF也有類似的faces-config.xml配置文件:
<faces-config> ?。糿avigation-rule> ?。糵rom-view-id>/index.jsp</from-view-id> ?。糿avigation-case> ?。糵rom-outcome>login</from-outcome> ?。紅o-view-id>/welcome.jsp</to-view-id> ?。?navigation-case> ?。?navigation-rule> ?。糾anaged-bean> ?。糾anaged-bean-name>user</managed-bean-name> ?。糾anaged-bean-class>com.corejsf.UserBean</managed-bean-class> ?。糾anaged-bean-scope>session</managed-bean-scope> ?。?managed-bean> </faces-config> |
在Struts-config.xml中有ActionForm Action以及Jsp之間的流程關系,在faces-config.xml中,也有這樣的流程,我們具體解釋一下Navigation:
在index.jsp中有一個事件:
<h:commandButton label="Login" action="login" /> |
action的值必須匹配form-outcome值,上述Navigation配置表示:如果在index.jsp中有一個login事件,那么事件觸發后下一個頁面將是welcome.jsp
JSF有一個獨立的事件發生和頁面導航的流程安排,這個思路比struts要非常清晰。
managed-bean類似Struts的ActionForm,正如可以在struts-config.xml中定義ActionForm的scope一樣,這里也定義了managed-bean的scope為session。
但是如果你只以為JSF的managed-bean就這點功能就錯了,JSF融入了新的Ioc模式/依賴性注射等技術。
查看關于
JSF,Struts 的全部文檔