深刻Struts2

剛剛看了一下,感受講的很清晰,這個對Java web有一點了解而後又想學習web

strut2的同窗 我想是很適合的了。apache

糾結,不知道本學期糾結的J2EE項目,該不應用 Struts2 ,session

無論了,學習一下以備不時之需。。app

1.    深刻Struts2的配置文件
本部分主要介紹struts.xml的經常使用配置。
1.1.    包配置:
Struts2框架中核心組件就是Action、攔截器等,Struts2框架使用包來管理Action和攔截器等。每一個包就是多個Action、多個攔截器、多個攔截器引用的集合。
在struts.xml文件中package元素用於定義包配置,每一個package元素定義了一個包配置。它的經常使用屬性有:
name:必填屬性,用來指定包的名字。
extends:可選屬性,用來指定該包繼承其餘包。繼承其它包,能夠繼承其它包中的Action定義、攔截器定義等。
namespace:可選屬性,用來指定該包的命名空間。
<! DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
        "http://struts.apache.org/dtds/struts-2.0.dtd" >
< struts >
    <!-- struts2 action 必須放在一個指定的包空間下定義 -->
    < package name = "default" extends = "struts-default" >
    <!-- 定義處理請求 URL login.action Action -->
        < action name = "login" class = "org.qiujy.web.struts.action.LoginAction" >
        <!-- 定義處理結果字符串和資源之間的映射關係 -->
            < result name = "success" > /success.jsp </ result >
            < result name = "error" > /error.jsp </ result >
        </ action >
    </ package >
</ struts >
如上示例的配置,配置了一個名爲default的包,該包下定義了一個Action。
1.2.    命名空間配置:
考慮到同一個Web應用中須要同名的Action,Struts2以命名空間的方式來管理Action,同一個命名空間不能有同名的Action。
Struts2經過爲包指定namespace屬性來爲包下面的全部Action指定共同的命名空間。
把上示例的配置改成以下形式:
<! DOCTYPE struts PUBLIC
        "-//Apache Software Foundation//DTD Struts Configuration 2.0//EN"
        "http://struts.apache.org/dtds/struts-2.0.dtd" >
< struts >
    <!-- struts2 action 必須放在一個指定的包空間下定義 -->
    < package name = "qiujy" extends = "struts-default" >
    <!-- 定義處理請求 URL login.action Action -->
        < action name = "login" class = "org.qiujy.web.struts2.action.LoginAction" >
        <!-- 定義處理結果字符串和資源之間的映射關係 -->
            < result name = "success" > /success.jsp </ result >
            < result name = "error" > /error.jsp </ result >
        </ action >
    </ package >
   
    < package name = "my" extends = "struts-default" namespace = "/manage" >
    <!-- 定義處理請求 URL login.action Action -->
        < action name = "backLogin" class = "org.qiujy.web.struts2.action.LoginAction" >
        <!-- 定義處理結果字符串和資源之間的映射關係 -->
            < result name = "success" > /success.jsp </ result >
            < result name = "error" > /error.jsp </ result >
        </ action >
    </ package > </ struts >
如上配置了兩個包:default和my,配置my包時指定了該包的命名空間爲/manage。
對於包default:沒有指定namespace屬性。若是某個包沒有指定namespace屬性,即該包使用默認的命名空間,默認的命名空間老是""。
對於包my:指定了命名空間/manage,則該包下全部的Action處理的URL應該是「命名空間/Action名」。如上名爲 backLogin的Action,它處理的URL爲:
http://localhost:8080/userlogin_struts2 /manage/backLogin.action
Struts2的命名空間的做用等同於struts1裏模塊的做用。
1.3.    包含配置:
在Struts2中能夠將一個配置文件分解成多個配置文件,那麼咱們必須在struts.xml中包含其餘配置文件。
< struts >
    < include file = "struts-default.xml" />
    < include file = "struts-user.xml" />
    < include file = "struts-book.xml" />
    < include file = "struts-shoppingCart.xml" />
   
    ......
   </ struts >
1.4.    攔截器配置:
見後面章節介紹。
1.5.    常量配置:
Struts2框架有兩個核心配置文件,其中struts.xml文件主要負責管理應 用中的Action映射, 及Action處理結果和物理資源之間的映射關係。除此以外,Struts2框架還包含了一個struts.properties文件,該文件主義了 Struts2框架的大量常量屬性。但一般推薦也是在struts.xml文件中來配置這些常量屬性。
如:後面會講到Struts2的國際化,它的資源文件位置就用常量屬性來指定:
< struts >
    ......
    < constant name = "struts.custom.i18n.resources" value = "messages" />
</ struts >
表示指定了資源文件的放置在classes目錄下,基本名是messages,則在classes目錄下您就應該放置相似messages_zh_CN.properties,message_en.properties名的文件。
2.    Struts2的Action
2.1.    實現Action類:
Struts2中Action是核心內容,它包含了對用戶請求的處理邏輯,咱們也稱Action爲業務控制器。
Struts2中的Action採用了低侵入式的設計,Struts2不要求 Action類繼承任何的Struts2的基類或實現Struts2接口。(可是,咱們爲了方便實現Action,大多數狀況下都會繼承 com.opensymphony.xwork2.ActionSupport類,並重寫此類裏的public String execute() throws Exception方法。由於此類中實現了不少的實用接口,提供了不少默認方法,這些默認方法包括獲取國際化信息的方法、數據校驗的方法、默認的處理用戶 請求的方法等,這樣能夠大大的簡化Action的開發。)
Struts2中一般直接使用Action來封裝HTTP請求參數,因 此,Action類裏還應該包含與請求參數對應的屬性,而且爲屬性提供對應的getter和setter方法。(固然,Action類中還能夠封裝處理結 果,把處理結果信息看成一屬性,提供對應的getter和setter方法)
修改第一部分的用戶登陸示例:把Action改爲以下:
package org.qiujy.web.struts2.action;
 
import com.opensymphony.xwork2.ActionSupport;
 
/**
  * @author qiujy
  * @version 1.0
  */
public class LoginAction extends ActionSupport {
    private String userName ;
    private String password ;
   
    private String msg ; // 結果信息屬性
   
    /**
      * @return the msg
      */
    public String getMsg() {
        return msg ;
    }
    /**
      * @param msg the msg to set
      */
    public void setMsg(String msg) {
        this . msg = msg;
    }
    /**
      * @return the userName
      */
    public String getUserName() {
        return userName ;
    }
    /**
      * @param userName the userName to set
      */
    public void setUserName(String userName) {
        this . userName = userName;
    }
    /**
      * @return the password
      */
    public String getPassword() {
        return password ;
    }
    /**
      * @param password the password to set
      */
    public void setPassword(String password) {
        this . password = password;
    }
   
    /**
      * 處理用戶請求的 excute() 方法
      * @return 結果導航字符串
      * @throws Exception
      */
    public String execute() throws Exception{
       if ( "test" .equals( this . userName ) &&
"test" .equals( this . password )){
           msg = " 登陸成功,歡迎 " + this . userName ;
           return this . SUCCESS ;
       } else {
           msg = " 登陸失敗,用戶名或密碼錯 " ;
           return this . ERROR ;
       }
    }
}
往success.jsp和error.jsp頁面中添加 ${msg} EL表達式來顯示結果信息。則最終效果跟之前同樣。
2.2.    Action訪問Servlet API
Struts2中的Action並無和任何Servlet API耦合,這樣框架更具靈活性,更易測試。
可是,對於web應用的控制器而言,不訪問Servlet API幾乎是不可能的,例如跟蹤HTTP Session狀態等。Struts2框架提供了一種更輕鬆的方式來訪問Servlet API。Struts2中提供了一個ActionContext類(當前Action的上下文對象),經過這個類能夠訪問Servlet API。下面是該類中提供的幾個經常使用方法:
public static ActionContext getContext() :得到當前Action的ActionContext實例。
public Object get(Object key) :此方法相似於調用HttpServletRequest的getAttribute(String name)方法。
public void put(Object key, Object value) :此方法相似於調用HttpServletRequest 的setAttribute(String name, Object o)。
public Map getParameters() :獲取全部的請求參數。相似於調用HttpServletRequest對象的getParameterMap() 方法。
public Map getSession() :返回一個Map對象,該Map對象模擬了HttpSession實例。
public void setSession(Map session) : 直接傳入一個Map實例,將該Map實例裏的key-value對轉換成session的屬性名-屬性值對。
public Map getApplication() :返回一個Map對象,該對象模擬了該應用的ServletContext實例。
public void setApplication(Map application) :直接傳入一個Map實例,將該Map實例裏的key-value對轉換成application的屬性名-屬性值對。
修改以上用戶登陸驗證示例的Action類中的execute方法:
public String execute() throws Exception{
        if ( "test" .equals( this . userName ) && "test" .equals( this . password )){
            msg = " 登陸成功,歡迎 " + this . userName ;
            // 獲取 ActionContext 實例,經過它來訪問 Servlet API
            ActionContext context = ActionContext.getContext();
            // session 中是否已經存放了用戶名,若是存放了:說明已經登陸了;
// 不然說明是第一次登陸成功
            if ( null != context.getSession().get( "uName" )){
                msg = this . userName + " :你已經登陸過了 !!!" ;
            } else {
                context.getSession().put( "uName" , this . userName );
            }
           
            return this . SUCCESS ;
        } else {
            msg = " 登陸失敗,用戶名或密碼錯 " ;
            return this . ERROR ;
        }
    }
       Struts2中經過ActionContext來訪問Servlet API,讓Action完全從Servlet API 中分離出來,最大的好處就是能夠脫離Web容器測試Action。
       另外,Struts2中還提供了一個ServletActionContext類,Action只要繼承自該類,就能夠直接訪問Servlet API。具體方法參看struts2的API文檔。
3.    一個Action內包含多個請求處理方法的處理
Struts1提供了DispatchAction,從而容許一個Action內包含多個請求處理方法。Struts2也提供了相似的功能。處理方式主要有如下三種方式:
3.1.    動態方法調用:
DMI:Dynamic Method Invocation 動態方法調用。
動態方法調用是指:表單元素的action不直接等於某個Action的名字,而是以以下形式來指定對應的動做名:
<form method="post" action="userOpt!login.action">
則用戶的請求將提交到名爲」userOpt」的Action實例,Action實例將調用名爲」login」方法來處理請求。同時login方法的簽名也是跟execute()同樣,即爲public String login() throws Exception。
注意:要使用動態方法調用,必須設置Struts2容許動態方法調用,經過設置struts.enable.DynamicMethodInvocation常量來完成,該常量屬性的默認值是true。
3.1.1.      示例:
修改用戶登陸驗證示例,多增長一個註冊用戶功能。
1.       修改Action類:
package org.qiujy.web.struts2.action;
 
import com.opensymphony.xwork2.ActionContext;
import com.opensymphony.xwork2.ActionSupport;
 
/**
  * @author qiujy
  * @version 1.0
  */
public class LoginAction extends ActionSupport{
    private String userName ;
    private String password ;
   
    private String msg ; // 結果信息屬性
   
    /**
      * @return the msg
      */
    public String getMsg() {
        return msg ;
    }
    /**
      * @param msg the msg to set
      */
    public void setMsg(String msg) {
        this . msg = msg;
    }
    /**
相關文章
相關標籤/搜索