Struts2 是一個用來開發(fā) MVC應用程序的框架. 它提供了 Web 應用程序開發(fā)過程中的一些常見問題的解決方案:
Struts2 = Struts1 + WebWork 1和2沒有本質的關系
1.Struts2開發(fā)流程
1.1 引入jar文件
commons-fileupload-1.2.2.jar 【文件上傳相關包】
commons-io-2.0.1.jar 【io操作相關的包】
struts2-core-2.3.4.1.jar 【struts2核心功能包】
xwork-core-2.3.4.1.jar 【Xwork核心包】
ognl-3.0.5.jar 【Ognl表達式功能支持表】
commons-lang3-3.1.jar 【struts對java.lang包的擴展】
freemarker-2.3.19.jar 【struts的標簽模板庫jar文件】
javassist-3.11.0.GA.jar 【struts對字節(jié)碼的處理相關jar】
1.2 配置web.xml
tomcat服務器在啟動時怨喘,首先會加載自身的web.xml,然后加載所有項目的web.xml。struts2通過在web.xml中引入過濾器來作為入口酬荞,在學習過濾器的時候我們知道過濾器的init方法會在tomcat服務器啟動時調用铸题。
<!-- 引入struts核心過濾器 -->
<filter>
<filter-name>struts2</filter-name>
<filter-class>org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>struts2</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
1.3 編寫Action類
action類耘拇,也叫做動作類; 一般繼承ActionSupport類,即處理請求的類 (struts中的action類取代之前的servlet).
注意: action中的業(yè)務方法不能有參數睦尽,且必須返回String
import com.opensymphony.xwork2.ActionSupport;
// 開發(fā)action: 處理請求
public class HelloAction extends ActionSupport {
// 處理請求
public String execute() throws Exception {
System.out.println("訪問到了action缺狠,正在處理請求");
return "success";
}
}
1.4 配置struts.xml
該配置文件的頭部可以參照struts2-core-2.3.4.1.jar包下面的struts-default.xml來寫问慎。該配置文件的主要作用是將提供action的訪問名稱,并且和action類進行關聯挤茄。
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE struts PUBLIC
"-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
"http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
<package name="xxxx" extends="struts-default">
<action name="hello" class="cn.acamy.action.HelloAction" method="execute">
<result name="success">/success.jsp</result>
</action>
</package>
</struts>
2.Struts2執(zhí)行流程
2.1 服務器啟動
- 加載項目web.xml
- 創(chuàng)建Struts核心過濾器對象如叼, 執(zhí)行filter的init()
struts-default.xml, 核心功能的初始化
struts-plugin.xml, struts相關插件
struts.xml 用戶編寫的配置文件
2.2 用戶訪問
3.用戶訪問Action, 服務器根據訪問路徑名稱,找對應的aciton配置, 創(chuàng)建action對象
4.執(zhí)行默認攔截器棧中定義的18個攔截器
5.執(zhí)行action的業(yè)務處理方法
6.根據action返回的Result來跳轉到相應的頁面
3. 三大配置文件的加載
從上面可以知道核心過濾器是Struts的入口穷劈,tomcat服務器在啟動時會執(zhí)行過濾器的init()方法笼恰,現在通過源碼展示三大配置文件是如何加載的。
3.1 首先是執(zhí)行web.xml引入的StrutsPrepareAndExecuteFilter初始化方法
public void init(FilterConfig filterConfig) throws ServletException {
InitOperations init = new InitOperations();
try {
FilterHostConfig config = new FilterHostConfig(filterConfig);
init.initLogging(config);
Dispatcher dispatcher = init.initDispatcher(config);// 在這里跳轉
init.initStaticContentLoader(config, dispatcher);
prepare = new PrepareOperations(filterConfig.getServletContext(), dispatcher);
execute = new ExecuteOperations(filterConfig.getServletContext(), dispatcher);
this.excludedPatterns = init.buildExcludedPatternsList(dispatcher);
postInit(dispatcher, filterConfig);
} finally {
init.cleanup();
}
}
3.2 然后執(zhí)行InitOperations的initDispatcher方法
public Dispatcher initDispatcher( HostConfig filterConfig ) {
Dispatcher dispatcher = createDispatcher(filterConfig);
dispatcher.init();
return dispatcher;
}
3.3 最后執(zhí)行Dispatcher的init方法
/**
* Load configurations, including both XML and zero-configuration strategies,
* and update optional settings, including whether to reload configurations and resource files.
*/
public void init() {
if (configurationManager == null) {
configurationManager = createConfigurationManager(BeanSelectionProvider.DEFAULT_BEAN_NAME);
}
try {
init_DefaultProperties(); // [1]
init_TraditionalXmlConfigurations(); // [2]跳轉到三大配置文件的相關方法
init_LegacyStrutsProperties(); // [3]
init_CustomConfigurationProviders(); // [5]
init_FilterInitParameters() ; // [6]
init_AliasStandardObjects() ; // [7]
Container container = init_PreloadConfiguration();
container.inject(this);
init_CheckConfigurationReloading(container);
init_CheckWebLogicWorkaround(container);
if (!dispatcherListeners.isEmpty()) {
for (DispatcherListener l : dispatcherListeners) {
l.dispatcherInitialized(this);
}
}
} catch (Exception ex) {
if (LOG.isErrorEnabled())
LOG.error("Dispatcher initialization failed", ex);
throw new StrutsException(ex);
}
}
//三大配置文件
private static final String DEFAULT_CONFIGURATION_PATHS = "struts-default.xml,struts-plugin.xml,struts.xml";
private void init_TraditionalXmlConfigurations() {
String configPaths = initParams.get("config");
if (configPaths == null) {
configPaths = DEFAULT_CONFIGURATION_PATHS;
}
String[] files = configPaths.split("\\s*[,]\\s*");
//驟個初始化
for (String file : files) {
if (file.endsWith(".xml")) {
if ("xwork.xml".equals(file)) {
configurationManager.addContainerProvider(createXmlConfigurationProvider(file, false));
} else {
configurationManager.addContainerProvider(createStrutsXmlConfigurationProvider(file, false, servletContext));
}
} else {
throw new IllegalArgumentException("Invalid configuration file name");
}
}
}
4.struts-default.xml, 詳解
位置:struts2-core-2.3.4.1.jar/ struts-default.xml
內容:
1. bean節(jié)點指定struts在運行的時候創(chuàng)建的對象類型
2.指定struts-default包 【用戶寫的package(struts.xml)一樣要繼承此包 】
package struts-default 包中定義了:
a. 跳轉的結果類型
dispatcher 轉發(fā)歇终,不指定默認為轉發(fā)
redirect 重定向
redirectAction 重定向到action資源
stream (文件下載的時候用)
b. 定義了所有的攔截器
定義了32個攔截器社证!
為了攔截器引用方便,可以通過定義棧的方式引用攔截器评凝,
此時如果引用了棧猴仑,棧中的攔截器都會被引用!
defaultStack
默認的棧,其中定義默認要執(zhí)行的18個攔截器肥哎!
c. 默認執(zhí)行的攔截器棧、默認執(zhí)行的action
<default-interceptor-ref name="defaultStack"/>
<default-class-ref class="com.opensymphony.xwork2.ActionSupport" />
<interceptor name="prepare"
class="com.opensymphony.xwork2.interceptor.PrepareInterceptor"/>
<interceptor name="params"
class="com.opensymphony.xwork2.interceptor.ParametersInterceptor"/>
注意:攔截器功能與過濾器功能類似疾渣,都攔截資源篡诽,但它是struts的概念,只能在struts中用榴捡,只攔截action請求;而過濾器是servlet的概念杈女,可以在struts項目、servlet項目用吊圾,可以攔截所有的web資源(/index.jsp/servlet/action/img/css/js)达椰。