Struts2的簡單使用


一、准備工作及實例

1.解壓struts-2.1.6-all.zip

apps目錄:struts2自帶的例子程序
docs目錄:官方文檔。

lib 目錄:存放所有jar文件。

Src 目錄:源文件存放地

2.六個基本包

struts2-core-2.1.6.jar 開發的核心類庫
freemarker-2.3.13.jar struts2的UI標簽的模板使用freemarker編寫
commons-logging-1.0.4.jar 日志包
ognl-2.6.11.jar
對象圖導航語言,通過它來讀寫對象屬性
xwork-2.1.2.jar xwork類庫,struts2在其上進行構建
commons-fileupload-1.2.1.jar
文件上傳組件,2.1.6版本后必須加入此jar包

特別需要說明的是目前strust2的最新版本是struts-2.1.6,它作為2.1X的正式版。特別要注意導入commons-fileupload-1.2.1.jar包,在此jar包中包含了RequestContext類,如果不導入該jar包將會報異常。

3.初識struts2配置文件

(1).web.xml文件   需要配置過濾器

主要完成對StrutsPrepareAndExecuteFilter的配置(在以前的版本中是對FilterDispatcher配置,新版本同樣支持用FilterDispatcher配置),它的實質是一個過濾器,它負責初始化整個Struts框架並且處理所有的請求。這個過濾器可以包括一些初始化參數,有的參數指定了要加載哪些額外的xml配置文件,還有的會影響struts框架的行為。除了StrutsPrepareAndExecuteFilter外,Struts還提供了一個ActionContexCleanUp類,它的主要任務是當有其它一些過濾器要訪問一個初始化好了的struts框架的時候,負責處理一些特殊的清除任務。

(2).struts.xml文件

框架的核心配置文件就是這個默認的struts.xml文件,在這個默認的配置文件里面我們可以根據需要再包括其它一些配置文件。在通常的應用開發中,我們可能想為每個不同的模塊單獨配置一個struts.xml文件,這樣也利於管理和維護。這也是我們要配置的主要文件。

(3).struts.properties(參default.properties)

在Struts框架使用了很多屬性,我們可以通過改變這些屬性來滿足我們的需求。要改變這些屬性,只需在struts.properties文件中指定屬性的key和value即可。屬性文件可以放在任何一個包含在classpath中的路徑上,但是通常我們都把它放在/WEB-INF/classes目錄下面。我們可以在struts-default.properties文件中找到一個屬性的列表。

(4)struts-default.xml

此文件是struts2框架默認加載的配置文件,它定義了struts2一些核心bean和攔截器,它會自動包含(included)到struts.xml文件中(實質是通過<package  extends="struts-default">),並為我們提供了一些標准的配置。我們可以在struts2-core.jar中找到這個文件。

(5)其它配置文件

velocity.properties,struts-default.vm,struts-plugin.xml

4.讓MyEclipse提示xml信息

當我們在編寫struts.xml時,發現eclipse並不會給出幫助提示,那是因為MyEclipse默認並不支持struts2,所以我們需要手工導入dtd以支持提示。步驟:[window][preferences][MyEclipse][Files and Editors][XML][xml Catelog]然后在右邊點add添加:location為dtd文件所在的位置(struts-2.0.dtd文件struts2-core-2.1.6.jar中可以得到),KeyType選擇URI,Key為struts-2.0.dtd文件中文檔聲明的內容(http://struts.apache.org/dtds/struts-2.0.dtd),在struts.xml文件中也有此key值。

5.如何使用alt+/提示

在MyEclipse6.5中,默認的提示為Ctrl+Space,而它會與我們的輸入法切換沖突,使提示失效。找到key,先取消Content Assist命令的綁定,再用“alt+/”來綁定。

6.實例

步驟一,新建myStruts2項目,並導入struts2的六個基本jar包。
步驟二,建立LoginAction文件,主要代碼如下:
package com.asm;

import com.opensymphony.xwork2.Action;

public class LoginAction implements Action {

    private String username;

    private String password;

...省略get/set方法

    public String execute() throws Exception {

        if (username.equals("struts2")) {

            return "loginSuccess";

        } else {

            return "loginFailure";

        }

    }

}
說明:實現了Action接口,主要是為了保證execute的正確定義,其實我們也可以不實現此接口,只要能保證execute方法書寫的正確書寫(方法名,返回值)。

步驟三,在struts.xml文件中注冊LoginAction。此配置文件要放在src目錄下,實質就是成為classpath環境變量下的文件。主要代碼如下:
<?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="myFirst" namespace="/" extends="struts-default">

        <action name="login" class="com.asm.LoginAction">

            <result name="loginSuccess">/success.jsp</result>

            <result name="loginFailure">/failure.jsp</result>

        </action>

    </package>

</struts>


說明package后面會有詳細說明。action元素中的name屬性值指定了此action所指定的請求路徑為“login.action”。后面login.jsp中的<form action=...>屬性值就會參照此name屬性。

步驟四、提供jsp頁面
login.jsp主要代碼:
<body>

        <form action="<%=request.getContextPath()%>/login.action" method="get">

        戶名:<input type="text" name="username"><br>

        密碼:<input type="password" name="password"><br>

        <input type="submit" value="login">

        </form>
</body>
failure.jsp主要代碼

<%@ page language="java" import="java.util.*" pageEncoding="UTF-8"%>

<%@ taglib uri="/struts-tags" prefix="s" %>
<html>

    <body>

        登錄失敗,錯誤的用戶名:<s:property value="username"/><br>

        <a href="<%=request.getContextPath()%>/login.jsp">返回</a>

    </body>

</html>
說明:使用了標簽庫,在struts2中使用標簽庫非常簡單,只需要像上面那樣導入標簽庫便可以使用所有的struts2的所有標簽

success.jsp主要代碼

<body> 登錄成功!</body>

步驟五、配置web.xml。完成核心監聽器注冊。內容如下:
<?xml version="1.0" encoding="UTF-8"?>

<web-app id="WebApp_9" 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">

    <filter>

        <filter-name>struts2</filter-name>

        <!-- <filter-class>org.apache.struts2.dispatcher.FilterDispatcher</filter-class>

        -->

        <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>

</web-app>

說明:注釋掉的部分為以前2.1.4版本中用的核心filter類。StrutsPrepareAndExecuteFilter類的init方法將會讀取類路徑下默認的配置文件struts.xml,並以javabean形式存放在內存中,以后struts2對用戶的每次請求將使用內存中數據,而不是重讀struts.xml文件。

步驟六、發布測試。
簡要分析執行流程
當輸入.../login.jsp訪問jsp頁面填寫完相關信息並提交給login.action時,它會首先被在web.xml中配置的過濾器監聽到,過濾器會去查找strust.xml文件,並結合namespace查找名為login的action,查找到此action便交給其處理,LoginAction內部會執行execute方法,並返回結果result(result也是參照的struts.xml中action下的result配置)。 關於表單傳參,主要是參照的action中的方法名,而非屬性名。

7.開啟struts2自帶的開發模式常量

在以前的開發中,當修改一些配置時總是不能及時地更新到服務器,我們總會重新部署或重啟來更新改變的內容,在struts2中可以通過一個常量來達到此目的。即在struts.xml中的<struts>元素下增加如下內容:<constant name="struts.configuration.xml.reload" value="true" /> 這樣配置后,當配置文件修改保存時就會及時更新到服務器中。其它一些常量:

<!-- 指定WEB應用的編碼集,相當於調用HttpServletRequest.setCharacterEncodint方法,如果使用了velocity或freemarker,它也用於指定輸出的編碼格式 -->

    <constant name="struts.i18n.encoding" value="UTF-8" />

    <!-- 指定請求后綴為.action,指定多個請求后綴用逗號分隔 -->

    <constant name="struts.action.extension" value="action" />

    <!--設置瀏覽器是否緩存靜態內容,建議:開發階段關閉,運行時開啟  -->

    <constant name="struts.serve.static.browserCache" value="false" />

    <!--當struts.xml配置文件修改后,系統是否重新加載該文件,開發階段打開此功能  -->

    <constant name="struts.configuration.xml.reload" value="true" />

    <!--  開發提示:出錯時打印更詳細的信息-->

    <constant name="struts.devMode" value="true" />

<!-- 指定請求的后綴可以是.do或.action -->

    <constant name="struts.action.extension" value="do,action" />

注意:在struts2.1.6版本中存在一個bug:即配置了struts.i18n.encoding常量也不能解決中文亂碼問題,原因是此版本在獲取請求參數后才調用了setCharacterEncoding()方法進行編碼設置。解決此bug的方法是配置一個filter,並在doFilter方法中增加如下代碼:request.setCharacterEncoding(“UTF-8”); 在以后的2.1.8版本中解決了此問題及2.1.6中存在的其它bug,建議新項目使用2.1.8版本。

8.vo傳參模式

Copy上面的myStruts2項目,改名為myStruts2Vo項目。作如下修改:在LoginAction中有兩個字段:username,password。把此兩個屬性重構到com.asm.vo.User類中,然后在LoginAction中提供User對象及相應的get/set方法。現在需要注意的是在login.jsp中會有如下的修改:

戶名:<input type="text" name="user.username"><br>

密碼:<input type="password" name="user.password"><br>

關鍵就是改掉name屬性值。其它基本無變動。 后話:假如此此User對象並不能和Model層的相應對象完全對應,我們還應借助此User對象在Action中構建出Model層的相應對象,這樣,在exectue方法中便能通過構建的Model對象作為參數與Model層交互。

9.ModerDriven傳參模式

Copy上面的myStruts2Vo項目,改名為myStruts2Model項目。重點是修改LoginAction,修改后的主要內容如下:
package com.asm;

import com.asm.vo.User;

import com.opensymphony.xwork2.Action;

import com.opensymphony.xwork2.ModelDriven;

public class LoginAction implements Action, ModelDriven<User> {

    private User user = new User();

    public String execute() throws Exception {

        if (user.getUsername().equals("struts2")) {

            return "loginSuccess";

        } else {

            return "loginFailure";

        }

    }

    public User getModel() {

        return user;

    }

}

說明:它實現了ModelDriven接口,並使用了泛性機制(必須),因此要求jdk1.5以上。
現在需要注意的是在login.jsp中name屬性值為User中兩個字段,和第一個實例一樣。說明:此方式一般不會使用,在此略作了解。

10.為什么要使用struts2代替struts1.x

(1)struts2的execute方法中的參數不會依賴於servletAPI,實現了也servlet解耦,是一種無侵入式的設計。

(2)struts2提供了攔截器,利用攔截器可以進行AOP編程,實現權限攔截等功能。

(3)struts2提供了類型轉換器,我們可以很容易地對請求參數轉換成需要的類型。

(4)提供了同種表現層技術支持,如JSP、freeMarker、velocity等

(5)可以對指定的方法進行校驗,可以輕松地實現表單校驗功能

(6)提供了全局范圍、包范圍和action范圍的國際化資源文件管理實現。

二、struts.xml配置及例程

1.配置文件的優先級

在struts2中一些配置(比如常量)可以同時在struts-default.xml(只讀性),strtus-plguin.xml(只讀性),struts.xml,struts.properties和web.xml文件中配置,它們的優先級逐步升高,即是說后面的配置會覆蓋掉前面相同的配置。

2.配置形式

下面以對struts.i18n.encoding=UTF-8的配置為例進行說明:

在struts.xml配置形式如下:

    <constant name="struts.i18n.encoding" value="gbk"></constant>

在struts.properties的配置形式如下:

struts.i18n.encoding=UTF-8

在web.xml中配置如下:
<filter>

<filter-name>struts2</filter-name>

<filter-class>    org.apache.struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter

    </filter-class>

    <init-param>

        <param-name>struts.i18n.encoding</param-name>

        <param-value>UTF-8</param-value>

    </init-param>

</filter>
說明:官方聲稱配置了此常量可以解決中文亂碼問題,但實事上並不能達到目的,在前面的三個項目中,如果我們在表單中輸入中文,其結果是會出現亂碼。解決此問題參看[一.7的注意]。這是struts2.1.6中的一bug,它的下一版2.1.8已解決此問題。

3.package配置相關

屬性名

是否必須

說明

Name

Package的唯一標識,不允許同名

Extends

指定要繼承的包

Namespace

指定名稱空間

Abstract

聲明包為抽象否

下面我們建立struts2package項目來進行package相關測試:

說明:在上面的配置文件中所用到的Test1Action和Test2Action這兩個Action都只是繼承了com.opensymphony.xwork2.ActionSupport類,而ActionSupport默認返回的就是“success”,所以當點擊上面的鏈接分別轉到了forward目錄下的test1.jsp和test2.jsp。下面重點來看這個package元素的namespace屬性及action的name屬性,它們共同定義了action所映射到的實質文件。上圖展示了鏈接地址和action的對應關系,所以當我們要想訪問一個action所關聯到的jsp文件時,應該用namespace+action的name 關於它的內容測試可以參考struts2package項目。
補充:通常情況下,action元素的name是屬性值是不能出現“/”的,所以希望通過action中name屬性來實現多級映射,需要在sturts.xml中增加如下屬性: 

<constant name="struts.enable.SlashesInActionNames" value="true"/> 這樣配置后就可以再action的name元素中使用“/”了。比如:
<package name="tt3" extends="struts-default">

        <action name="test3/test3" class="com.asm.Test3Action">

            <result name="success">/forward/test3.jsp</result>

        </action>

</package>

然后輸入<a href="<%=path%>/test3/test3.action">test3</a><br>鏈接地址就可以訪問了

強調:namespace默認值“”,即不配置namespace屬性。它的意思是:如果action不能進行完整路徑匹配,則會來此namespace下進行匹配,比如:.../test/test/test.action,如果參照namespace及action的name不能找到也之完全對應的action,它會再到依次追溯到上級目錄中查找,即是說它會以…/test/test.action這樣的路徑來對應namespace和action的name進行查找。如果返回到最終的目錄仍找不到,它就會到namespace="/"對應的包下查找名為test的action,如果仍找不到,它就會去默認的namespace下查找名為test的action,如果找到則執行此action。另外,namespace也可以配置成namespace="/"。它代表配置為項目的根。 總結action的名稱探索順序:完全對應、逐步追溯到上級目錄查找、"/"下查找、默認namespace下查找。

為什么要提出namespace,主要是避免多人共同開發項目出現名字沖突。如果不使用namespace,多個人所寫的action中可能出現重名的現象,這樣當項目合並時就會出現沖突。而有了namespace可以在項目開發時由項目經理給每一個人分不同的namespace,這樣每個開發人員只需要保證自己所寫的action不同名即可。
namespace引發的鏈接問題:當我們為action配置了namespace時,訪問此action的形式總會是如下形式:.../webappname/xxx/yyy/ActionName.action 而當此action成功執行跳轉到某個jsp頁面時,如想在此jsp頁面寫鏈接,一定要寫絕對路徑,因為相對路徑是相對.../webappname/xxx/yyy/,而如果以后我們修改了action的namespace時,相對路徑又要變,所以鏈接不能寫成相對路徑。 以下介紹絕對路徑的寫法:通常用myeclipse開發時建立一個jsp文件,默認總會有如下內容:

<%

String path = request.getContextPath();

String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";

%>
我們寫絕對路徑可以參此內容。還可以參<head>下的<base href="<%=basePath%>"> 來完成絕對路徑的書寫。

4.分工合作include:指定多個配置文件

比如讓jack來單獨開發一個action,在jack.xml中的配置文件為:
<struts>

    <package name="jack" namespace="/jack" extends="struts-default">

        <action name="test4" class="com.asm.Test4Action">

            <result name="success">/forward/test4.jsp</result>

        </action>

    </package>

</struts>

然后在struts.xml文件中增加如下內容:<include file="jack.xml"></include> 它實質就是把jack.xml中的<package>及其內容寫進struts.xml中的<struts>根元素下。

鏈接:<a href="<%=path %>/jack/test4.action">test4</a> 這樣便可以訪問到了forward目錄下的test4.jsp了。

5.tomcat認證訪問

接上例:namespce的作用除了在前面提到的避免協同開發名字沖突外,還為認證提供一個條件。比如jack開發的東西所關聯到的頁面需要權限才能被訪問。由於多為tomcat中的內容,下面只列出步驟。
步驟一,tomcat的conf目錄下tomcat-users.xml內容如下:
<?xml version='1.0' encoding='utf-8'?>

<tomcat-users>

  <role rolename="manager"/>

  <role rolename="admin"/>

  <user username="jack" password="jack" roles="admin,manager"/>

  <user username="tom" password="tom" roles="manager"/>

</tomcat-users>
步驟二,在web.xml中增加如下內容:
<security-constraint>

        <web-resource-collection>

            <web-resource-name>jack</web-resource-name>

            <url-pattern>/jack/*</url-pattern>

            <http-method>POST</http-method>

            <http-method>GET</http-method>

        </web-resource-collection>

        <auth-constraint>

            <role-name>admin</role-name>

        </auth-constraint>

    </security-constraint>

 

    <security-role>

        <role-name>admin</role-name>

    </security-role>

 

    <login-config>

        <auth-method>BASIC</auth-method>

        <realm-name>input authentication message</realm-name>

    </login-config>

這樣配置完成后,當我們訪問.../jack中的任何內容都會要求輸入密碼認證信息,認證時輸入tomcat-users.xml配置的admin權限的用戶名和密碼即可訪問(這里就只有jack用戶名可以訪問)

6.初識攔截器

攔截器能在action被調用之前和被調用之后執行一些“代碼”。Struts2框架的大部分核心功能都是通過攔截器來實現的,如防止重復提交、類型轉換、對象封裝、校驗、文件上傳、頁面預裝載等等,都是在攔截器的幫助下實現的。每一個攔截器都是獨立裝載的(pluggable),我們可以根據實際的需要為每一個action配置它所需要的攔截器。

myStruts2項目下,重新對配置文件作如下修改:
<package name="myFirst" namespace="/" extends="struts-default">

<interceptors>

<interceptor name="timer"

                class="com.opensymphony.xwork2.interceptor.TimerInterceptor" />


    <interceptor name="params"                                                                                                        class="com.opensymphony.xwork2.interceptor.ParametersInterceptor" />

    </interceptors>

 

        <action name="login" class="com.asm.LoginAction">

            <interceptor-ref name="timer"></interceptor-ref>

            <interceptor-ref name="params"></interceptor-ref>

            <result name="loginSuccess">/success.jsp</result>

            <result name="loginFailure">/failure.jsp</result>

        </action>

    </package>

首先在package中定義了兩個攔截器,然后在login action中引用了這兩個攔截器,需要說明的是這里使用的攔截器都是系統自帶的攔截器。其實在extends所繼承的struts-default中就包含了很多攔截器,也包括我們這里所用的攔截器,但如果在此action中不使用params攔截器,將會報空指針錯,因為params攔截器的作用是傳遞表單參數,如果不使用此攔截器就不能在action中得到表單參數,所以引用時會報空指針錯。雖然extends繼承的strust-default自帶有params攔截器,但是當我們自己引用了攔截器時,繼承struts-default將不會再為我們分配默認的攔截器(有點類似構造器),但是我們仍然可以通過<interceptor-ref name="defaultStack"/>來繼續使用struts-defalut的攔截器。補充:由於上面的package繼承於struts-default,而我們這里所用到的timer和params都是在struts-defalut中定義過,所以即使我們在<interceptors>中沒有定義過這兩個攔截器,也可以直接在action中引用。

使用</interceptor-stack>組合多個攔截器:比如我們想把上面的params和timer這兩個攔截器組合:

    <interceptor-stack name="timer_param">

                <interceptor-ref name="timer" />

                <interceptor-ref name="params" />

    </interceptor-stack>

然后再在action引用<interceptor-ref name="timer_param"/>”,效果和分別引用兩個是一樣的。其實我們使用strtus-default中的<interceptor-ref name="defaultStack"/>也是使用interceptor-stack方式。

7.Action中的method屬性

在struts1.x中我們知道通過繼承DispatchAction可以實現把多個Action進行統一操作,在struts2中實現action的統一操作也很簡單。我們以crud操作為例,把crud集中到一個Action中。
步驟一、建立CRUDAction,內容如下:
package com.asm;

import com.opensymphony.xwork2.ActionSupport;

public class CRUDAction extends ActionSupport {

    public String add() {

        return "success";

    }

    public String del() {

        return "success";

    }

    public String update() {

        return "success";

    }

    public String query() {

        return "success";

    }

}

步驟二、配置此Action,為了清晰明了,專為此Action,建立一個配置文件crud.xml,主要內容如下:
<struts>

    <package name="crud" extends="struts-default" namespace="/crud">

        <action name="add" class="com.asm.CRUDAction" method="add">

            <result name="success">/crud/addSuccess.jsp</result>

        </action>

        <action name="del" class="com.asm.CRUDAction" method="del">

            <result name="success">/crud/delSuccess.jsp</result>

        </action>

        <action name="update" class="com.asm.CRUDAction" method="update">

            <result name="success">/crud/updateSuccess.jsp</result>

        </action>

        <action name="query" class="com.asm.CRUDAction" method="query">

            <result name="success">/crud/querySuccess.jsp</result>

        </action>

    </package>

</struts>

分析:上面的method方法的值來源於CRUDAction中方法的名字,這樣當我們訪問上面的每一個Action時,它實質是和method指定的方法關聯上。

步驟三、把crud.xml配置文件並入struts.xml中,只需增加如下代碼:
<include file="jack.xml"></include>

步驟四、編寫相應的jsp頁面,在此略去crud文件夾下的四個跳轉jsp頁面(addSuccess.jsp等),重點是crud.jsp頁面。內容如下:
<html>

<%

    String path=request.getContextPath();

%>

    <body>

        <a href="<%=path %>/crud/add.action">添加數據</a><br>

        <a href="<%=path %>/crud/del.action">刪除數據</a><br>

        <a href="<%=path %>/crud/query.action">查詢數據</a><br>

        <a href="<%=path %>/crud/update.action">修改數據</a><br>

    </body>

</html>
步驟五、發布測試。
補充擴展,動態調用DMI不使用method實現統一.我們在crud.xml中增加如下內容:
<action name="op" class="com.asm.CRUDAction">

            <result name="success">/crud/op.jsp</result>

</action>

然后再在crud.jsp中定義如下鏈接:

<a href="<%=path %>/crud/op!add.action">添加數據</a><br>

<a href="<%=path %>/crud/op!del.action">刪除數據</a><br>

<a href="<%=path %>/crud/op!query.action">查詢數據</a><br>

<a href="<%=path %>/crud/op!update.action">修改數據</a><br>

注意查看上面的鏈接地址,它們都是針對op action,然后再加地上“!+CRUDAction中相應的方法名”,最后再寫上.action即可以訪問到統一頁面op.jsp。這樣做雖然能減少頁面,但是由於它們實質用到的是同一個Action,所以這就意味着我們要使用的攔截器相同,相同的跳轉result。實際中這種方式很少使用,在此略作了解。如果不想使用動態方法調用,我們可以通過常量來關閉,即在struts.xml中增加如下配置:
<constant name="struts.enable.DynamicMethodInvocation" value="false"/>

擴展2:在CRUDAction中使用do。舉例:我們在CRUDAction中增加一個新的方法,內容如下:

    public String doMain(){

        return "success";

    }

然后再在在crud.xml中增加如下內容:

<action name="main" class="com.asm.CRUDAction" method="main">

            <result name="success">/crud/main.jsp</result>

</action>
注意:配置中method屬性值是doMain中去掉do后M小寫。然后再在crud.jsp中增加如下鏈接:
<a href="<%=path %>/crud/main.action">main頁面</a><br>

隨后便可以訪問到.../crud/main.jsp頁面了。

8.使用ForwardAction實現頁面屏蔽。

我們在jsp頁面之間寫鏈接總會是.../xxx.jsp,而如果我們想屏蔽掉具體的jsp,只需要所jsp頁面配置成一個ForwardAction即可實現。示例如下:在根目錄下有一個index.jsp主頁,我們strtus.xml中作如下配置:

<package name="def" extends="struts-default">    

        <action name="forward">

            <result >/index.jsp</result>

        </action>

</package>

說明:如果沒有未action指定class,默認就是ActionSupport類,如果沒有為action指定method屬性,則默認執行execute方法,如果沒有指定result的name屬性,默認值為success。知道了這些再結合ActionSupport的源碼就不難理解實現轉發的原理了。
隨后再在前面第7點擴展中用到的op.jsp中增加如下代碼:
<a href="<%=request.getContextPath()%>/forward.action">forward</a>

最后再測試訪問op.jsp,在op.jsp中頁面中直接點鏈接便可以跳到index.jsp,觀察地址欄發現此時跳到index頁面是進行的服務器跳轉,如果我們在上面的配置中的result增加type屬性變成<result type="redirect">/index.jsp</result>,實現的跳轉就是客戶端跳轉。 補充:像這種forward形式的action實質是執行的ActionSupport 這個Action。因此配置它的result可以參看此類的api文檔,比如它常用的result name有:success、login、input等。

8.使用default-Action配置統一訪問

default-action-ref,當訪問沒有找到對應的action時,默認就會調用default-action-ref指定的action.同樣在上面的package中增加如下內容:

<default-action-ref name="error"></default-action-ref>

        <action name="error">

            <result>/other/error.jsp</result>

        </action>

上面一段內容就是說當我們訪問的action不能被找到時便指向名為error的action中去,接着我們在下面配置了這個error Action。但是要注意,一個package內只配置一個<default-action-ref>,如果配置多個,就無法預測結果了. 此時我們只要輸入.../myStruts2/luanFangWen.action這樣的形式,它就會去訪問這個默認的<default-action-ref>,通常我們會為它配置一個錯誤頁面,以提示用戶訪問的頁面不存在。 在web開發中,我們還可以把這個默認的action訪問配置成主頁,這樣當用戶訪問一些不存在的action時,總會跳到主頁上去。

通過此配置,只要是訪問一個不存在的action便會轉向到.../other目錄下的error.jsp頁面。但是如果訪問是其它的不存在資源則仍是報tomcat所標識的404錯誤,我們可以在web.xml中作如下配置:
<error-page>

        <error-code>404</error-code>

        <location>/other/404error.jsp</location>

</error-page>

這樣配置后,訪問錯誤頁面將跳到.../other/404error.jsp頁面中去。補充說明:如果我們用ie訪問時,如果選中了[工具][IE選項][高級][瀏覽][顯示友好的http錯誤信息],則配置的錯誤頁面將失效,因為找不到資源時會報HTTP404錯誤,而ie截取到此錯誤進行了它自身的友好處理,所以我們設置<error-page>就失效。

小結Action

在struts2中一個普通的java類只要有public String execute()這樣的方法都可以配置成一個Action,另外我們可以實現Action接口來使java類成為一個Action,但通常的做法是繼承ActionSupport類,這也是以后的項目中慣用的方法,也是推薦的首選方法。 與struts1.x不同的是:在struts2中每一個Action被請求訪問時都會new出這個Action對象,所以Action本身不存在線程安全的問題。

9.使用通配符

建立struts2wildcard項目,此實例基本仿照前面前面第7點的實例改寫而成。為了使用通配符,只需要改寫配置文件即可。此實例未使用通配時的配置文件如下:

<action name="addUser" class="com.asm.UserAction" method="addUser">

            <result name="success">/user/addUser.jsp</result>
</action>

<action name="delUser" class="com.asm.UserAction" method="delUser">

            <result name="success">/user/delUser.jsp</result>

</action>

<action name="queryUser" class="com.asm.UserAction" method="queryUser">

            <result name="success">/user/queryUser.jsp</result>

</action>

<action name="updateUser" class="com.asm.UserAction" method="updateUser">

           <result name="success">/user/updateUser.jsp</result>

</action>

我們注釋掉上面的配置,使用通配符只需如下內容即可達到相同的效果:
<action name="*User" class="com.asm.UserAction" method="{1}User">

            <result name="success">/user/{1}User.jsp</result>

</action>

原理:當有.../addUser.action請求時,如果不能在當前應用中找到完全相同的addUser名字的Action時,通配符配置這時就起作用了,按通配原則,它便和上面的name="*User"相配成功,這里不難明了*此時代指的內容是add,再來看method恰恰是引用第一個*的內容,所以它的method此時的完整名為addUser,它剛好和com.asmUserAction中的addUser方法相對,所以它會去addUser方法,再來看下面的result配置所指代的頁面,它也用到了{1},所以它的完整頁面是/addUser.jsp。其實如果我們有良好的編程命名習慣,所有的Action我們都只需要進行一次配置。舉例:規定所有的Action類都用XXXAction來命名,類中所有的CRUD方法都用add/del/update/query。Jsp頁面也用add/del/update/query_XXX.jsp這樣的形式。即配置文件可以寫成如下形式:
<action name="*_*" class="com.asm.{2}Action" method="{1}">

            <result name="success">.../{1}_{2}.jsp</result>

</action>

Name中第一個*代表CRUD操作的名字,第二個*代表類的名字。所以訪問鏈接地址舉例如下:

.../del_User.action將訪問到User類的del方法,成功后跳到del_User.jsp頁面。補充說明{0}是代表name中所有的*組合。

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM