ActionContext和ServletActionContext小結

一隻小螞蟻吆發表於2014-08-21

1. ActionContext

在Struts2開發中,除了將請求引數自動設定到Action的欄位中,我們往往也需要在Action裡直接獲取請求(Request)或會話(Session)的一些資訊,甚至需要直接對JavaServlet Http的請求(HttpServletRequest),響應(HttpServletResponse)操作. 我們需要在Action中取得request請求引數"username"的值:

ActionContext context = ActionContext.getContext();
Map params = context.getParameters();
String username = (String) params.get("username");
ActionContext(com.opensymphony.xwork.ActionContext)是Action執行時的上下文,上下文可以看作是一個容器(其實我們這裡的容器就是一個Map而已),它存放的是Action在執行時需要用到的物件. 一般情況, 我們的ActionContext都是通過: ActionContext context = (ActionContext) actionContext.get();來獲取的.我們再來看看這裡的actionContext物件的建立:

static ThreadLocal actionContext = new ActionContextThreadLocal();

ActionContextThreadLocal是實現ThreadLocal的一個內部類.ThreadLocal可以命名為"執行緒區域性變數",它為每一個使用該變數的執行緒都提供一個變數值的副本,使每一個執行緒都可以獨立地改變自己的副本,而不會和其它執行緒的副本衝突.這樣,我們ActionContext裡的屬性只會在對應的當前請求執行緒中可見,從而保證它是執行緒安全的.

通過ActionContext取得HttpSession: Map session = ActionContext.getContext().getSession();

  2. ServletActionContext

ServletActionContext(com.opensymphony.webwork. ServletActionContext),這個類直接繼承了我們上面介紹的ActionContext,它提供了直接與Servlet相關物件訪問的功能,它可以取得的物件有:

(1)javax.servlet.http.HttpServletRequest : HTTPservlet請求物件

(2)javax.servlet.http.HttpServletResponse : HTTPservlet相應物件

(3)javax.servlet.ServletContext : Servlet上下文資訊

(4)javax.servlet.ServletConfig : Servlet配置物件

(5)javax.servlet.jsp.PageContext : Http頁面上下文

如何從ServletActionContext裡取得Servlet的相關物件:

<1>取得HttpServletRequest物件: HttpServletRequest request = ServletActionContext. getRequest();

<2>取得HttpSession物件: HttpSession session = ServletActionContext. getRequest().getSession();

  3. ServletActionContext和ActionContext聯絡

ServletActionContext和ActionContext有著一些重複的功能,在我們的Action中,該如何去抉擇呢?我們遵循的原則是:如果ActionContext能夠實現我們的功能,那最好就不要使用ServletActionContext,讓我們的Action儘量不要直接去訪問Servlet的相關物件.

注意:在使用ActionContext時有一點要注意: 不要在Action的建構函式裡使用ActionContext.getContext(),因為這個時候ActionContext裡的一些值也許沒有設定,這時通過ActionContext取得的值也許是null;同樣,HttpServletRequest req = ServletActionContext.getRequest()也不要放在建構函式中,也不要直接將req作為類變數給其賦值。至於原因,我想是因為前面講到的static ThreadLocal actionContext = new ActionContextThreadLocal(),從這裡我們可以看出ActionContext是執行緒安全的,而ServletActionContext繼承自ActionContext,所以ServletActionContext也執行緒安全,執行緒安全要求每個執行緒都獨立進行,所以req的建立也要求獨立進行,所以ServletActionContext.getRequest()這句話不要放在建構函式中,也不要直接放在類中,而應該放在每個具體的方法體中(eg:login()、queryAll()、insert()等),這樣才能保證每次產生物件時獨立的建立了一個req。

  4. struts2中獲得request、response和session

(1)非IoC方式

方法一:使用org.apache.struts2.ActionContext類,通過它的靜態方法getContext()獲取當前Action的上下文物件。

ActionContext ctx = ActionContext.getContext();

ctx.put("liuwei", "andy"); //request.setAttribute("liuwei", "andy");
Map session = ctx.getSession(); //session

HttpServletRequest request = ctx.get(org.apache.struts2.StrutsStatics.HTTP_REQUEST);
HttpServletResponse response = ctx.get(org.apache.struts2.StrutsStatics.HTTP_RESPONSE);
細心的朋友可以發現這裡的session是個Map物件, 在Struts2中底層的session都被封裝成了Map型別. 我們可以直接操作這個Map物件進行對session的寫入和讀取操作, 而不用去直接操作HttpSession物件.

方法二:使用org.apache.struts2.ServletActionContext類

public class UserAction extends ActionSupport {
    
    //其他程式碼片段
    
    private HttpServletRequest req;
// private HttpServletRequest req = ServletActionContext.getRequest(); 這條語句放在這個位置是錯誤的,同樣把這條語句放在構造方法中也是錯誤的。

    public String login() {
        req = ServletActionContext.getRequest(); //req的獲得必須在具體的方法中實現
        user = new User();
        user.setUid(uid);
        user.setPassword(password);
        if (userDAO.isLogin(user)) {
            req.getSession().setAttribute("user", user);
            return SUCCESS;
        }
        return LOGIN;
    }
    public String queryAll() {
        req = ServletActionContext.getRequest(); //req的獲得必須在具體的方法中實現
        uList = userDAO.queryAll();
        req.getSession().setAttribute("uList", uList);
        return SUCCESS;
    }
    
    //其他程式碼片段
}

(2)IoC方式(即使用Struts2 Aware攔截器)

要使用IoC方式,我們首先要告訴IoC容器(Container)想取得某個物件的意願,通過實現相應的介面做到這點。

public class UserAction extends ActionSupport implements SessionAware, ServletRequestAware, ServletResponseAware {

    private HttpServletRequest request;
    private HttpServletResponse response;

    public void setServletRequest(HttpServletRequest request) {
        this.request = request;
    }

    public void setServletResponse(HttpServletResponse response) {
        this.response = response;
    }

    public String execute() {
        HttpSession session = request.getSession();
        return SUCCESS;
    }
}

<script type="text/javascript"></script>

相關文章