Struts2 與 Struts1.x的深度比較

 
Struts做爲MVC 2的Web框架,自推出以來不斷受到開發者的追捧,獲得普遍的應用。做爲最成功的Web框架,Struts天然擁有衆多的優勢:MVC 2模型的使用、功能齊全的標誌庫(Tag Library)、開放源代碼。
可是,正所謂「沒有最好,只有更好」,Struts1.x自身也有很多的缺點:須要編寫的代碼過多,容易引發「類爆炸」、單元測試困難。這些缺點隨着Web的發展愈來愈明顯。這就促生了Struts 2,它的誕生能很好的解決上述問題。
在本文中,筆者將對Struts2和Struts1.x這兩種框架進行詳細的比較。比較將涉及到這兩種框架的Action、驗證、類型轉換及如何開發等方面的內容。但願經過這樣的比較,讓讀者瞭解這兩種框架各自的特色,以便於在本身的項目中,根據實際狀況,儘快的過渡到Struts2的時代。本文的內容基於Struts2.0.6。
1、引言
Struts的第一個版本是在2001年5月份發佈的。它的最初設想是經過結合 JSP和Servlet,使Web應用的視圖和業務/應用邏輯得以清晰地分離開來。在Struts以前,最多見的作法是在JSP中加入業務和應用邏輯,或者在Servlet中經過println()來生成視圖。
自從初版發佈以來,Struts實際上已成爲業界公認的Web應用標準。它的煊赫一時也爲本身帶來了改進和變動,因此不但要跟上對Web應用框架不斷變化的需求,並且要與日漸增多競爭激烈的衆多框架的特性相融合。
到最後,產生了幾個下一代Struts的解決方案。其中兩個最受矚目的方案是 Shale和Struts Ti。Shale是一個基於構件的框架,並在最近成爲Apache的頂級項目。而Struts Ti則是在Struts的成功經驗基礎上繼續堅持對前端控制器(Front Controller)和MVC(model-view-controller)模式進行改進。
WebWork項目是在2002年3月發佈的,它對Struts式框架進行了革命性改進,引進了很多新的思想、概念和功能,但和原Struts代碼並不兼容。WebWork是一個成熟的框架,通過了好幾回重大的改進與發佈。
在2005年12月,WebWork與Struts Ti宣佈合併。與此同時,Struts Ti更名爲Struts Action Framework 2.0,成爲Struts真正的繼承者。
最後要注意的是,並非說Struts或WebWork項目已經中止開發了。因爲人們對這兩個項目的興趣仍然很高,並且也有不少開發者仍然願意使用它們,所以這兩個項目還在繼續開發中,繼續修復Bug,改進功能和繼續添加新功能。
2、 Action的區別
對於有着豐富的Struts1.x開發經驗的朋友來講,都十分的清楚Action是整個Struts框架的核心內容,固然Struts2也不例外。不過,Struts1.x與Struts2的Action模型很大的區別。
Struts2和Struts1.x的差異,最明顯的就是Struts2是一個 pull-MVC架構。這是什麼意思呢?從開發者角度看,就是說須要顯示給用戶的數據能夠直接從Action中獲取,而不像Struts1.x那樣,必須把相應的Bean存到Page、Request或者Session中才能獲取。Struts1.x 必須繼承org.apache.struts.action.Action或者其子類,表單數據封裝在FormBean中。Struts 2無須繼承任何類型或實現任何接口,表單數據包含在Action中,經過Getter和Setter獲取(以下面的ActionForStruts2的代碼示例)。
雖然,在理論上Struts2的Action無須實現任何接口或者是繼承任何的類,可是,在實際編程過程當中,爲了更加方便的實現Action,大多數狀況下都會繼承 com.opensymphony.xwork2.ActionSupport類,而且重載(Override)此類裏的String execute()方法。以下所示:
        
        
        
        
         
         
         
         package ActionDiffer;
 import java.text.DateFormat;
 import java.util.Date;
 import com.opensymphony.xwork2.ActionSupport;
 public class ActionForStruts2 extends ActionSupport {
     private String message;
     public String getMessage() {
         return message;
    } 
    @Override 
     public String execute() {
    message = " This is hello from strtuts2. Now is: " + 
DateFormat.getInstance().format( new Date());
         return SUCCESS;
    } 
}
        
        
        
        
首先,從ActionForStruts2能夠看出,返回的對象不是 ActionForward,而是String。若是你不喜歡以字符串的形式出如今你的代碼中,有個Helper接口Action能夠以常量方式提供常見結果,如「success」、「none」、「error」、「input」和「login」。
另外, 按照慣例,在Struts1.x中只有「execute」方法能調用Action, 但在Struts2中並不是必要,任何聲明爲public String methodName() 方法,都能經過配置來調用Action。
最後,和Struts1.x最大的革命性的不一樣是,Struts2處理Action 過程當中調用的方法(「execute」方法)是不帶參數的。那如何獲取所須要的對象呢?答案是使用IoC(反轉控制,Inversion of Control),也叫「依賴注入(Dependency Injection)」的模式(想更多地瞭解這方面信息請看Martin Fowler的文章[url]http://www.martinfowler.com/articles/injection.html[/url])。Spring框架使得這個模式流行起來,然而Struts2的前身(WebWork)也同時應用上了這個模式。
3、IoC
IoC(Inversion of Control,如下譯爲控制反轉),隨着Java社區中輕量級容器(Lightweight Contianer)的推廣而愈來愈爲你們耳熟能詳。在此,無需再多費脣舌來解釋「什麼是控制反轉」和「爲何須要控制反轉」。由於互聯網上已經有很是多的文章對諸如此類的問題做了精彩而準確的回答。讀者能夠去讀一下Rod Johnson和Juergen Hoeller合著的《Expert one-on-one J2EE Development without EJB》或Martin Fowler所寫的《Inversion of Control Containers and the Dependency Injection pattern》。
衆所周知,Struts2是以Webwork 2做爲基礎發展出來。而在Webwork 2.2以前的Webwork版本,其自身有一套控制反轉的實現,Webwork 2.2在Spring 框架的如火如荼發展的背景下,決定放棄控制反轉功能的開發,轉由Spring實現。值得一提的是,Spring確實是一個值得學習的框架,由於有愈來愈多的開源組件(如iBATIS等)都放棄與Spring重疊的功能的開發。所以,Struts2推薦你們經過Spring實現控制反轉。
爲了更好地瞭解反轉控制,下面來看一個例子,如何利用IoC在Action處理過程當中能夠訪問到當前請求HttpServerRequest對象。
在例子中,使用的依賴注入機制是接口注入。就如其名稱同樣,接口注入須要的是已經被實現了的接口。這個接口包含了相應屬性的setter,爲Action提供值。例子中使用了ServletRequestAware接口,以下:
        
        
        
        
         
         
         
         public interface ServletRequestAware {
    public void setServletRequest(HttpServletRequest request);
}
        
        
        
        
當繼承這個接口後,本來簡單的Action看起來有點複雜了,可是這時能夠獲取HttpServerRequest對象來使用了。
        
        
        
        
         
         
         
         public class IoCForStruts2 implements ServletRequestAware {
   private HttpServletRequest request;
   public void setServletRequest(HttpServletRequest request) {
        this.request = request;
   }
   public String execute() throws Exception {
        // 能夠開始使用request對象進行工做了
        return Action.SUCCESS;
   }
}
        
        
        
        
看起來如今這些屬性是類級別的,並非線程安全的,會出現問題。其實在Struts2裏並無問題,由於每一個請求過來的時候都會產生一個新的Action對象實例,它並無和其餘請求共享一個對象,因此不須要考慮線程安全問題。
4、攔截器
Interceptor(如下譯爲攔截器),在AOP(Aspect-Oriented Programming)中用於在某個方法或字段被訪問以前,進行攔截而後在以前或以後加入某些操做。攔截是AOP的一種實現策略。
在Webwork的中文文檔的解釋爲——攔截器是動態攔截Action調用的對象。它提供了一種機制可使開發者定義在一個action執行的先後執行的代碼,也能夠在一個action執行前阻止其執行。同時也提供了一種能夠提取action中可重用的部分的方式。
Struts1.x的標準框架中不提供任何形式的攔截器,雖一個名爲SAIF的附加項目則實現了這樣的功能,但它的適用的範圍還頗有限。
攔截器是Struts2的一個強有力的工具,有許多功能(feature)都是構建於它之上,如國際化、轉換器,校驗等。談到攔截器,還有一個流行的詞——攔截器鏈(Interceptor Chain,在Struts2中稱爲攔截器棧Interceptor Stack)。攔截器鏈就是將攔截器按必定的順序聯結成一條鏈。在訪問被攔截的方法或字段時,攔截器鏈中的攔截器就會按其以前定義的順序被調用。
Struts 2的攔截器實現相對比較簡單。當請求到達Struts2的ServletDispatcher時,Struts 2會查找配置文件,並根據其配置實例化相對的攔截器對象,而後串成一個列表(list),最後一一地調用列表中的攔截器。
Struts 2已經提供豐富多樣功能齊全的攔截器實現。讀者能夠到struts2-all-2.0.6.jar或struts2-core-2.0.6.jar包的struts-default.xml查看關於默認的攔截器與攔截器鏈的配置。
做爲「框架(framework)」,可擴展性是不可缺乏的,由於世上沒有放之四海而皆準的東西。雖然,Struts 2爲咱們提供如此豐富的攔截器實現,可是這並不意味咱們失去建立自定義攔截器的能力,偏偏相反,在Struts 2自定義攔截器是至關容易的一件事。
5、Struts2和Struts1.x的全面比較
爲了對Struts2和Strtus1.x進行全面的比較,讓讀者瞭解這兩種框架各自的優缺點,以便於在本身的項目中,根據實際狀況,選擇合適的框架,對它們二者進行比較,總結了以下表分析比較。
特性比較:
Action類
Struts1.x要求Action類要擴展自一個抽象基類。Struts1.x的一個共有的問題是面向抽象類編程而不是面向接口編程。
Struts2的Action類實現了一個Action接口,連同其餘接口一塊兒來實現可選擇和自定義的服務。Struts2提供一個名叫ActionSupport的基類來實現通常使用的接口。固然,Action接口不是必須的。任何使用execute方法的POJO對象能夠被看成Struts 2的Action對象來使用。
線程模型
Struts1.x Action類是單例類,由於只有一個實例來控制全部的請求。單例類策略形成了必定的限制,而且給開發帶來了額外的煩惱。Action資源必須是線程安全或者同步的。
Struts2 Action對象爲每個請求都實例化對象,因此沒有線程安全的問題。(實踐中,servlet容器給每個請求產生許多丟棄的對象,而且不會致使性能和垃圾回收問題)。
Servlet 依賴
Struts1.x的Action類依賴於servlet API,當Action被調用時,以HttpServletRequest和HttpServletResponse做爲參數傳給execute方法。
Struts2的Action和容器無關。Servlet上下文被表現爲簡單的 Maps,容許Action被獨立的測試。Struts2的Action能夠訪問最初的請求(若是須要的話)。可是,儘量避免或排除其餘元素直接訪問 HttpServletRequest或HttpServletResponse。
易測性
測試Struts1.x的主要問題是execute方法暴露了Servlet API這使得測試要依賴於容器)。第三方的擴展,如Struts TestCase,提供了一套Struts1的模擬對象(來進行測試)。
Struts2的Action能夠經過初始化、設置屬性、調用方法來測試。依賴注入的支持也是測試變得更簡單。
捕獲輸入
Struts1.x使用ActionForm對象來捕獲輸入。象Action同樣,全部的ActionForm必須擴展基類。由於其餘的JavaBean不能做爲ActionForm使用,開發者常常建立多餘的類來捕獲輸入。 DynaBeans能夠被用來做爲替代ActionForm的類來建立。可是,開發者多是在從新描述(建立)已經存在的JavaBean(仍然會致使有冗餘的javabean)。
Struts2直接使用Action屬性做爲輸入屬性,消除了對第二個輸入對象的需求。輸入屬性多是有本身(子)屬性的rich對象類型。Action屬性可以經過web頁面上的taglibs訪問。Struts2也支持 ActionForm模式。rich對象類型,包括業務對象,可以用做輸入/輸出對象。這種ModelDriven 特性簡化了taglib對POJO輸入對象的引用。
表達式語言
Struts1.x整合JSTL,因此它使用JSTL的表達式語言。表達式語言有基本的圖形對象移動,可是對集合和索引屬性的支持很弱。
Struts2使用JSTL,可是也支持一個更強大和靈活的表達式語言--"Object Graph Notation Language" (OGNL)。
將值綁定到頁面
Struts1.x使用標準JSP機制來綁定對象到頁面上下文。
Struts2使用「ValueStack」技術,使taglib可以訪問值而不須要把你的頁面(view)和對象綁定起來。ValueStack策略容許經過一系列名稱相同但類型不一樣的屬性重用頁面(view)。
類型轉換
Struts1.x的ActionForm屬性常常都是String。Struts 1.x使用Commons-Beanutils來進行類型轉換。轉換每個類,而不是爲每個實例配置。
Struts2使用OGNL進行類型轉換。提供基本和經常使用對象的轉換器。
驗證
Struts1.x支持在ActionForm的validate方法中手動校驗,或者經過Commons Validator的擴展來校驗。同一個類能夠有不一樣的校驗內容,但不能校驗子對象。
Struts2支持經過validate方法和XWork校驗框架來進行校驗。XWork校驗框架使用爲屬性類類型定義的校驗和內容校驗,來支持chain校驗子屬性
Action執行控制
Struts1.x支持每個模塊有單獨的Request Processors(生命週期),可是模塊中的全部Action必須共享相同的生命週期。
Struts2支持經過攔截器堆棧(Interceptor Stacks)爲每個Action建立不一樣的生命週期。堆棧可以根據須要和不一樣的Action一塊兒使用。
6、結論
前面已經簡要介紹了Struts2的起源,並詳細對比了Struts2和 Struts1.x的差別,讀者應該對Struts2的基礎有所瞭解了——包括高層的框架概念和基礎的請求流程,並理解Struts1.x和 Struts2二者之間在Action方面的差異,Struts2增強了對攔截器與IoC的支持,而在Struts1.x中,這些特性是很難想象的。
同時,讀者應該明白:Struts2是WebWork的升級,而不是Struts 1.x的升級。雖然Struts 2提供了與Struts1.x的兼容,但已經不是Struts1.x的升級。對於已有Struts1.x開發經驗的開發者而言,Struts1.x的開發經驗對於Struts2並無太大的幫助;相反,對於已經有WebWork開發經驗的開發者而言,WebWork的開發經驗對Struts2的開發將有很好的借鑑意義
相關文章
相關標籤/搜索