IoC(Inversion of Control,以下譯為控制反轉(zhuǎn))隨著Java社區(qū)中輕量級(jí)容器(Lightweight Contianer)的推廣而越來(lái)越為大家耳熟能詳。在此,我不想再多費(fèi)唇舌來(lái)解釋“什么是控制反轉(zhuǎn)”和“為什么需要控制反轉(zhuǎn)”。因?yàn)榛ヂ?lián)網(wǎng)上已經(jīng)有非常多的文章對(duì)諸如此類的問(wèn)題作了精彩而準(zhǔn)確的回答。大家可以去讀一下Rod Johnson和Juergen Hoeller合著的《Expert one-on-one J2EE Development without EJB》或Martin Fowler所寫(xiě)的《Inversion of Control Containers and the Dependency Injection pattern》。
言歸正傳,本文的目的主要是介紹在Struts 2中實(shí)現(xiàn)控制反轉(zhuǎn)。
歷史背景
眾所周知,Struts 2是以Webwork 2作為基礎(chǔ)發(fā)展出來(lái)。而在Webwork 2.2之前的Webwork版本,其自身有一套控制反轉(zhuǎn)的實(shí)現(xiàn),Webwork 2.2在Spring 框架的如火如荼發(fā)展的背景下,決定放棄控制反轉(zhuǎn)功能的開(kāi)發(fā),轉(zhuǎn)由Spring實(shí)現(xiàn)。值得一提的是,Spring確實(shí)是一個(gè)值得學(xué)習(xí)的框架,因?yàn)橛性絹?lái)越多的開(kāi)源組件(如iBATIS等)都放棄與Spring重疊的功能的開(kāi)發(fā)。因此,Struts 2推薦大家通過(guò)Spring實(shí)現(xiàn)控制反轉(zhuǎn)。
具體實(shí)現(xiàn)
首先,在開(kāi)發(fā)環(huán)境中配置好Struts 2的工程。對(duì)這部分仍然有問(wèn)題的朋友,請(qǐng)參考我的早前的文章。
然后,將所需的Spring的jar包加入到工程的構(gòu)建環(huán)境(Build Path)中,如下圖1所示:
圖1 所依賴的Spring的jar包 本文使用的是Spring 2.0,Spring強(qiáng)烈建議大家在使用其jar包時(shí),只引用需要的包,原因是Spring是一個(gè)功能非常強(qiáng)大的框架,其中有些功能是您不需要的;而且Spring提倡的是“按需所取”,而不是EJB的“愛(ài)我就要愛(ài)我的一切”。當(dāng)然,如果你怕麻煩或者是不清楚每個(gè)包的作用,引用一個(gè)Spring的總包也未嘗不可。
接下來(lái),就要修改WEB-INF\web.xml文件了,內(nèi)容為:
<? xml version="1.0" encoding="UTF-8" ?>
< web-app version ="2.4" xmlns ="http://java.sun.com/xml/ns/j2ee"
xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation ="http://java.sun.com/xml/ns/j2ee
http://java.sun.com/xml/ns/j2ee/web-app_2_4.xsd" >
< display-name > Struts 2 IoC Demo </ display-name >
< filter >
< filter-name > struts-cleanup </ filter-name >
< filter-class >
org.apache.struts2.dispatcher.ActionContextCleanUp
</ filter-class >
</ filter >
< filter >
< filter-name > struts2 </ filter-name >
< filter-class >
org.apache.struts2.dispatcher.FilterDispatcher
</ filter-class >
</ filter >
< filter-mapping >
< filter-name > struts-cleanup </ filter-name >
< url-pattern > /* </ url-pattern >
</ filter-mapping >
< filter-mapping >
< filter-name > struts2 </ filter-name >
< url-pattern > /* </ url-pattern >
</ filter-mapping >
< listener >
< listener-class >
org.springframework.web.context.ContextLoaderListener
</ listener-class >
</ listener >
< welcome-file-list >
< welcome-file > index.html </ welcome-file >
</ welcome-file-list >
</ web-app >
清單1 WEB-INF\web.xml大家一看便知道,主要是加入Spring的ContextLoaderListener監(jiān)聽(tīng)器,方便Spring與Web容器交互。
緊接著,修改Struts.properties文件,告知Struts 2運(yùn)行時(shí)使用Spring來(lái)創(chuàng)建對(duì)象(如Action等),內(nèi)容如下:
struts.objectFactory = spring
清單2 classes\struts.properties再下來(lái),遵循Spring的原則——面向接口編程,創(chuàng)建接口ChatService,代碼如下:
package tutorial;
import java.util.Set;
public interface ChatService {
Set < String > getUserNames();
} 清單3 tutorial.ChatService.java然后,再創(chuàng)建一個(gè)默認(rèn)實(shí)現(xiàn)ChatServiceImpl,代碼如下:
package tutorial;
import java.util.HashSet;
import java.util.Set;
public class ChatServiceImpl implements ChatService {
public Set < String > getUserNames() {
Set < String > users = new HashSet < String > ();
users.add( " Max " );
users.add( " Scott " );
users.add( " Bob " );
return users;
}
} 清單4 tutorial.ChatServiceImpl.java接下來(lái),就該新建Action了。tutorial.ChatAction.java的代碼如下:
package tutorial;
import java.util.Set;
import com.opensymphony.xwork2.ActionSupport;
public class ChatAction extends ActionSupport {
private static final long serialVersionUID = 8445871212065L ;
private ChatService chatService;
private Set < String > userNames;
public void setChatService(ChatService chatService) {
this .chatService = chatService;
}
public Set < String > getUserNames() {
return userNames;
}
@Override
public String execute() {
userNames = chatService.getUserNames();
return SUCCESS;
}
} 清單5 tutorial.ChatAction.javaChatAction類使用屬性(Getter/Setter)注入法取得ChatService對(duì)象。
然后,配置Spring的applicationContext.xml(位于WEB-INF下)文件,內(nèi)容如下:
<? xml version="1.0" encoding="UTF-8" ?>
< beans xmlns ="http://www.springframework.org/schema/beans"
xmlns:xsi ="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation ="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-2.0.xsd" >
< bean id ="chatService" class ="tutorial.ChatServiceImpl" />
< bean id ="chatAction" class ="tutorial.ChatAction" scope ="prototype" >
< property name ="chatService" >
< ref local ="chatService" />
</ property >
</ bean >
</ beans >
清單6 WEB-INF\applicationContext.xml上述代碼有二點(diǎn)值得大家注意的:
- Struts 2會(huì)為每一個(gè)請(qǐng)求創(chuàng)建一個(gè)Action對(duì)象,所以在定義chatAction時(shí),使用scope="prototype"。這樣Spring就會(huì)每次都返回一個(gè)新的ChatAction對(duì)象了;
- 因?yàn)镃hatServiceImpl被配置為默認(rèn)的scope(也即是singleton,唯一的),所以在實(shí)現(xiàn)時(shí)應(yīng)保證其線程安全(關(guān)于編寫(xiě)線程安全的代碼的討論已經(jīng)超出本文的范圍,更超出了本人的能力范圍,大家可以參考Addison Wesley Professional出版的《Java Concurrency in Practice》)。
接下來(lái),在classes/struts.xml中配置Action,內(nèi)容如下:
<! DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
"http://struts.apache.org/dtds/struts-2.0.dtd" >
< struts >
< include file ="struts-default.xml" />
< package name ="Struts2_IoC" extends ="struts-default" >
< action name ="Chat" class ="chatAction" >
< result > /UserList.jsp </ result >
</ action >
</ package >
</ struts >
清單7 classes\struts.xml這里的Action和平常不同的就是class屬性,它對(duì)應(yīng)于Spring所定義的bean的id,而不是它的類全名。
最后,讓我們看看/UserList.jsp,內(nèi)容如下:
<% @ page contentType = " text/html; charset=UTF-8 " %>
<% @ taglib prefix = " s " uri = " /struts-tags " %>
< html >
< head >
< title > User List </ title >
</ head >
< body >
< h2 > User List </ h2 >
< ol >
< s:iterator value ="userNames" >
< li >< s:property /></ li >
</ s:iterator >
</ ol >
</ body >
</ html >
清單8 /UserList.jsp大功告成,分布運(yùn)行應(yīng)用程序,在瀏覽器中鍵入http://localhost:8080/Struts2_IoC/Chat.action,出現(xiàn)如圖2所示頁(yè)面:
圖2 /ListUser.jsp 總結(jié)
通過(guò)Spring在Struts 2上實(shí)現(xiàn)控制反轉(zhuǎn)是強(qiáng)烈推薦的做法,當(dāng)然您也可以組合其它的實(shí)現(xiàn)(如Pico等)。