使用 Spring 更好地處理 Struts 動作----三種整合 Struts 應用程式與 Spring 的方式

xuniji123發表於2007-01-03

Struts Recipes 的合著者 George Franciscus 將介紹另一個重大的 Struts 整合竅門 —— 這次是將 Struts 應用程式匯入 Spring 框架。請跟隨 George,他將向您展示如何改變 Struts 動作,使得管理 Struts 動作就像管理 Spring beans 那樣。結果是一個增強的 web 框架,這個框架可以方便地利用 Spring AOP 的優勢。

您肯定已經聽說過控制反轉 (IOC) 設計模式,因為很長一段時間以來一直在流傳關於它的資訊。如果您在任何功能中使用過 Spring 框架,那麼您就知道其原理的作用。在本文中,我利用這一原理把一個 Struts 應用程式注入 Spring 框架,您將親身體會到 IOC 模式的強大。

將一個 Struts 應用程式整合進 Spring 框架具有多方面的優點。首先,Spring 是為解決一些關於 JEE 的真實世界問題而設計的,比如複雜性、低效能和可測試性,等等。第二,Spring 框架包含一個 AOP 實現,允許您將面向方面技術應用於物件導向的程式碼。第三,一些人可能會說 Spring 框架只有處理 Struts 比 Struts 處理自己好。但是這是觀點問題,我演示三種將 Struts 應用程式整合到 Spring 框架的方法後,具體由您自己決定使用哪一種。

我所演示的方法都是執行起來相對簡單的,但是它們卻具有明顯不同的優點。我為每一種方法建立了一個獨立而可用的例子,這樣您就可以完全理解每種方法。請參閱 下載 部分獲得完整例子原始碼。請參閱 參考資料,下載 Struts MVC 和 Spring 框架。

Spring 的創立者 Rod Johnson 以一種批判的眼光看待 Java™ 企業軟體開發,並且提議很多企業難題都能夠透過戰略地使用 IOC 模式(也稱作依賴注入)來解決。當 Rod 和一個具有奉獻精神的開放原始碼開發者團隊將這個理論應用於實踐時,結果就產生了 Spring 框架。簡言之,Spring 是一個輕型的容器,利用它可以使用一個外部 XML 配置檔案方便地將物件連線在一起。每個物件都可以透過顯示一個 JavaBean 屬性收到一個到依賴物件的引用,留給您的簡單任務就只是在一個 XML 配置檔案中把它們連線好。

依賴注入是一個強大的特性,但是 Spring 框架能夠提供更多特性。Spring 支援可插拔的事務管理器,可以給您的事務處理提供更廣泛的選擇範圍。它整合了領先的永續性框架,並且提供一個一致的異常層次結構。Spring 還提供了一種使用面向方面程式碼代替正常的物件導向程式碼的簡單機制。

Spring AOP 允許您使用攔截器 在一個或多個執行點上攔截應用程式邏輯。加強應用程式在攔截器中的日誌記錄邏輯會產生一個更可讀的、實用的程式碼基礎,所以攔截器廣泛用於日誌記錄。您很快就會看到,為了處理橫切關注點,Spring AOP 釋出了它自己的攔截器,您也可以編寫您自己的攔截器。

與 Struts 相似,Spring 可以作為一個 MVC 實現。這兩種框架都具有自己的優點和缺點,儘管大部分人同意 Struts 在 MVC 方面仍然是最好的。很多開發團隊已經學會在時間緊迫的時候利用 Struts 作為構造高品質軟體的基礎。Struts 具有如此大的推動力,以至於開發團隊寧願整合 Spring 框架的特性,而不願意轉換成 Spring MVC。沒必要進行轉換對您來說是一個好訊息。Spring 架構允許您將 Struts 作為 Web 框架連線到基於 Spring 的業務和持久層。最後的結果就是現在一切條件都具備了。

在接下來的小竅門中,您將會了解到三種將 Struts MVC 整合到 Spring 框架的方法。我將揭示每種方法的缺陷並且對比它們的優點。 一旦您瞭解到所有三種方法的作用,我將會向您展示一個令人興奮的應用程式,這個程式使用的是這三種方法中我最喜歡的一種。

接下來的每種整合技術(或者竅門)都有自己的優點和特點。我偏愛其中的一種,但是我知道這三種都能夠加深您對 Struts 和 Spring 的理解。在處理各種不同情況的時候,這將給您提供一個廣闊的選擇範圍。方法如下:

  • 使用 Spring 的 ActionSupport 類整合 Structs
  • 使用 Spring 的 DelegatingRequestProcessor 覆蓋 Struts 的 RequestProcessor
  • 將 Struts Action 管理委託給 Spring 框架

無論您使用哪種技術,都需要使用 Spring 的 ContextLoaderPlugin 為 Struts 的 ActionServlet 裝載 Spring 應用程式環境。就像新增任何其他外掛一樣,簡單地向您的 struts-config.xml 檔案新增該外掛,如下所示:

  "org.springframework.web.struts.ContextLoaderPlugIn">
"contextConfigLocation" value="/WEB-INF/beans.xml"/>

前面已經提到過,在 下載 部分,您能夠找到這三個完全可使用的例子的完整原始碼。每個例子都為一個書籍搜尋應用程式提供一種不同的 Struts 和 Spring 的整合方法。您可以在這裡看到例子的要點,但是您也可以下載應用程式以檢視所有的細節。

手動建立一個 Spring 環境是一種整合 Struts 和 Spring 的最直觀的方式。為了使它變得更簡單,Spring 提供了一些幫助。為了方便地獲得 Spring 環境,org.springframework.web.struts.ActionSupport 類提供了一個 getWebApplicationContext() 方法。您所做的只是從 Spring 的 ActionSupport 而不是 Struts Action 類擴充套件您的動作,如清單 1 所示:

package ca.nexcel.books.actions;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.struts.action.ActionError;
import org.apache.struts.action.ActionErrors;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.action.DynaActionForm;
import org.springframework.context.ApplicationContext;
import org.springframework.web.struts.ActionSupport;

import ca.nexcel.books.beans.Book;
import ca.nexcel.books.business.BookService;

public class SearchSubmit extends ActionSupport { |(1)


public ActionForward execute(
ActionMapping mapping,
ActionForm form,
HttpServletRequest request,
HttpServletResponse response)
throws IOException, ServletException {

DynaActionForm searchForm = (DynaActionForm) form;
String isbn = (String) searchForm.get("isbn");

//the old fashion way
//BookService bookService = new BookServiceImpl();

ApplicationContext ctx =
getWebApplicationContext(); |(2)
BookService bookService =
(BookService) ctx.getBean("bookService"); |(3)

Book book = bookService.read(isbn.trim());

if (null == book) {
ActionErrors errors = new ActionErrors();
errors.add(ActionErrors.GLOBAL_ERROR,new ActionError
("message.notfound"));
saveErrors(request, errors);
return mapping.findForward("failure") ;
}

request.setAttribute("book", book);
return mapping.findForward("success");
}
}

讓我們快速思考一下這裡到底發生了什麼。在 (1) 處,我透過從 Spring 的 ActionSupport 類而不是 Struts 的 Action 類進行擴充套件,建立了一個新的 Action。在 (2) 處,我使用 getWebApplicationContext() 方法獲得一個 ApplicationContext。為了獲得業務服務,我使用在 (2) 處獲得的環境在 (3) 處查詢一個 Spring bean。

這種技術很簡單並且易於理解。不幸的是,它將 Struts 動作與 Spring 框架耦合在一起。如果您想替換掉 Spring,那麼您必須重寫程式碼。並且,由於 Struts 動作不在 Spring 的控制之下,所以它不能獲得 Spring AOP 的優勢。當使用多重獨立的 Spring 環境時,這種技術可能有用,但是在大多數情況下,這種方法不如另外兩種方法合適。

將 Spring 從 Struts 動作中分離是一個更巧妙的設計選擇。分離的一種方法是使用 org.springframework.web.struts.DelegatingRequestProcessor 類來覆蓋 Struts 的 RequestProcessor 處理程式,如清單 2 所示:



br /> "-//Apache Software Foundation//DTD Struts Configuration 1.1//EN"
"



type="org.apache.struts.validator.DynaValidatorForm">














type="ca.nexcel.books.actions.SearchSubmit"
input="/searchEntry.do"
validate="true"
name="searchForm">








|(1)


value="/WEB-INF/validator-rules.xml,/WEB-INF/validation.xml"/>







我利用了 標記來用 DelegatingRequestProcessor 覆蓋預設的 Struts RequestProcessor。下一步是在我的 Spring 配置檔案中註冊該動作,如清單 3 所示:


br /> "




class="ca.nexcel.books.actions.SearchSubmit"> |(1)




注意:在 (1) 處,我使用名稱屬性註冊了一個 bean,以匹配 struts-config 動作對映名稱。SearchSubmit 動作揭示了一個 JavaBean 屬性,允許 Spring 在執行時填充屬性,如清單 4 所示:

package ca.nexcel.books.actions;

import java.io.IOException;

import javax.servlet.ServletException;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;

import org.apache.struts.action.Action;
import org.apache.struts.action.ActionError;
import org.apache.struts.action.ActionErrors;
import org.apache.struts.action.ActionForm;
import org.apache.struts.action.ActionForward;
import org.apache.struts.action.ActionMapping;
import org.apache.struts.action.DynaActionForm;

import ca.nexcel.books.beans.Book;
import ca.nexcel.books.business.BookService;

public class SearchSubmit extends Action {

private BookService bookService;
public BookService getBookService() {
return bookService;
}

public void setBookService(BookService bookService) { | (1)
this.bookService = bookService;
}

public ActionForward execute(
ActionMapping mapping,
ActionForm form,
HttpServletRequest request,
HttpServletResponse response)
throws IOException, ServletException {

DynaActionForm searchForm = (DynaActionForm) form;
String isbn = (String) searchForm.get("isbn");

Book book = getBookService().read(isbn.trim()); |(2)

if (null == book) {
ActionErrors errors = new ActionErrors();
errors.add(ActionErrors.GLOBAL_ERROR,new ActionError("message.notfound"));
saveErrors(request, errors);
return mapping.findForward("failure") ;
}

request.setAttribute("book", book);
return mapping.findForward("success");
}

}

在清單 4 中,您可以瞭解到如何建立 Struts 動作。在 (1) 處,我建立了一個 JavaBean 屬性。DelegatingRequestProcessor自動地配置這種屬性。這種設計使 Struts 動作並不知道它正被 Spring 管理,並且使您能夠利用 Sping 的動作管理框架的所有優點。由於您的 Struts 動作注意不到 Spring 的存在,所以您不需要重寫您的 Struts 程式碼就可以使用其他控制反轉容器來替換掉 Spring。

DelegatingRequestProcessor 方法的確比第一種方法好,但是仍然存在一些問題。如果您使用一個不同的 RequestProcessor,則需要手動整合 Spring 的 DelegatingRequestProcessor。新增的程式碼會造成維護的麻煩並且將來會降低您的應用程式的靈活性。此外,還有過一些使用一系列命令來代替 Struts RequestProcessor 的傳聞。 這種改變將會對這種解決方法的使用壽命造成負面的影響。

一個更好的解決方法是將 Strut 動作管理委託給 Spring。您可以透過在 struts-config 動作對映中註冊一個代理來實現。代理負責在 Spring 環境中查詢 Struts 動作。由於動作在 Spring 的控制之下,所以它可以填充動作的 JavaBean 屬性,併為應用諸如 Spring 的 AOP 攔截器之類的特性帶來了可能。

清單 5 中的 Action 類與清單 4 中的相同。但是 struts-config 有一些不同:



br /> "-//Apache Software Foundation//DTD Struts Configuration 1.1//EN"
"



type="org.apache.struts.validator.DynaValidatorForm">














type="org.springframework.web.struts.DelegatingActionProxy" |(1)
input="/searchEntry.do"
validate="true"
name="searchForm">










property="pathnames"
value="/WEB-INF/validator-rules.xml,/WEB-INF/validation.xml"/>



className="org.springframework.web.struts.ContextLoaderPlugIn">




清單 5 是一個典型的 struts-config.xml 檔案,只有一個小小的差別。它註冊 Spring 代理類的名稱,而不是宣告動作的類名,如(1)處所示。DelegatingActionProxy 類使用動作對映名稱查詢 Spring 環境中的動作。這就是我們使用 ContextLoaderPlugIn 宣告的環境。

將一個 Struts 動作註冊為一個 Spring bean 是非常直觀的,如清單 6 所示。我利用動作對映使用 標記的名稱屬性(在這個例子中是 "/searchSubmit")簡單地建立了一個 bean。這個動作的 JavaBean 屬性像任何 Spring bean 一樣被填充:


br /> "




class="ca.nexcel.books.actions.SearchSubmit">





動作委託解決方法是這三種方法中最好的。Struts 動作不瞭解 Spring,不對程式碼作任何改變就可用於非 Spring 應用程式中。RequestProcessor 的改變不會影響它,並且它可以利用 Spring AOP 特性的優點。

動作委託的優點不止如此。一旦讓 Spring 控制您的 Struts 動作,您就可以使用 Spring 給動作補充更強的活力。例如,沒有 Spring 的話,所有的 Struts 動作都必須是執行緒安全的。如果您設定 標記的 singleton 屬性為“false”,那麼不管用何種方法,您的應用程式都將在每一個請求上有一個新生成的動作物件。您可能不需要這種特性,但是把它放在您的工具箱中也很好。您也可以利用 Spring 的生命週期方法。例如,當例項化 Struts 動作時, 標記的 init-method 屬性被用於執行一個方法。類似地,在從容器中刪除 bean 之前,destroy-method 屬性執行一個方法。這些方法是管理昂貴物件的好辦法,它們以一種與 Servlet 生命週期相同的方式進行管理。

前面提到過,透過將 Struts 動作委託給 Spring 框架而整合 Struts 和 Spring 的一個主要的優點是:您可以將 Spring 的 AOP 攔截器應用於您的 Struts 動作。透過將 Spring 攔截器應用於 Struts 動作,您可以用最小的代價處理橫切關注點。

雖然 Spring 提供很多內建攔截器,但是我將向您展示如何建立自己的攔截器並把它應用於一個 Struts 動作。為了使用攔截器,您需要做三件事:

  1. 建立攔截器。
  2. 註冊攔截器。
  3. 宣告在何處攔截程式碼。

這看起來非常簡單的幾句話卻非常強大。例如,在清單 7 中,我為 Struts 動作建立了一個日誌記錄攔截器。 這個攔截器在每個方法呼叫之前列印一句話:

package ca.nexcel.books.interceptors;

import org.springframework.aop.MethodBeforeAdvice;

import java.lang.reflect.Method;

public class LoggingInterceptor implements MethodBeforeAdvice {

public void before(Method method, Object[] objects, Object o) throws Throwable {
System.out.println("logging before!");
}
}

這個攔截器非常簡單。before() 方法在攔截點中每個方法之前執行。在本例中,它列印出一句話,其實它可以做您想做的任何事。下一步就是在 Spring 配置檔案中註冊這個攔截器,如清單 8 所示:


br /> "




class="ca.nexcel.books.actions.SearchSubmit">





<!-- Interceptors --&gt
class="ca.nexcel.books.interceptors.LoggingInterceptor"/> |(1)

<!-- AutoProxies --&gt
class="org.springframework.aop.framework.autoproxy.
BeanNameAutoProxyCreator"> |(2)

/searchSubmit|(3)



logger |(4)




您可能已經注意到了,清單 8 擴充套件了 清單 6 中所示的應用程式以包含一個攔截器。具體細節如下:

  • 在 (1) 處,我註冊了這個攔截器。
  • 在 (2) 處,我建立了一個 bean 名稱自動代理,它描述如何應用攔截器。還有其他的方法定義攔截點,但是這種方法常見而簡便。
  • 在 (3) 處,我將 Struts 動作註冊為將被攔截的 bean。如果您想要攔截其他的 Struts 動作,則只需要在 "beanNames" 下面建立附加的 標記。
  • 在 (4) 處,當攔截髮生時,我執行了在 (1) 處建立的攔截器 bean 的名稱。這裡列出的所有攔截器都應用於“beanNames”。

就是這樣。就像這個例子所展示的,將您的 Struts 動作置於 Spring 框架的控制之下,為處理您的 Struts 應用程式提供了一系列全新的選擇。在本例中,使用動作委託可以輕鬆地利用 Spring 攔截器提高 Struts 應用程式中的日誌記錄能力。

在本文中,您已經學習了將 Struts 動作整合到 Spring 框架中的三種竅門。使用 Spring 的 ActionSupport 來整合 Struts(第一種竅門中就是這樣做的)簡單而快捷,但是會將 Struts 動作與 Spring 框架耦合在一起。如果您需要將應用程式移植到一個不同的框架,則需要重寫程式碼。第二種解決方法透過委託 RequestProcessor 巧妙地解開程式碼的耦合,但是它的可擴充套件性不強,並且當 Struts 的 RequestProcessor 變成一系列命令時,這種方法就持續不了很長時間。第三種方法是這三種方法中最好的:將 Struts 動作委託給 Spring 框架可以使程式碼解耦,從而使您可以在您的 Struts 應用程式中利用 Spring 的特性(比如日誌記錄攔截器)。

三種 Struts-Spring 整合竅門中的每一種都被實現成一個完整可用的應用程式。

[@more@]

來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/8271432/viewspace-888219/,如需轉載,請註明出處,否則將追究法律責任。

相關文章