1. 問題
Struts2 的 Action 我們將它定義為一個控制器,但是由于在 Action 中也可以來編寫一些業(yè)務(wù)邏輯,也有人會在 Action 輸入業(yè)務(wù)邏輯層。
但是在企業(yè)開發(fā)中,我們一般會將業(yè)務(wù)邏輯層單獨編寫,而不是將它與 action 層寫到一起。
之前的練習(xí)中,我們一直將屬性如 username 、 password 等保存在了 action 中。
這樣做了以后導(dǎo)致我們在調(diào)用業(yè)務(wù)邏輯層時可能需要將Action的對象傳過去。
但是這樣做無異于直接將 Servlet 傳給 Service,導(dǎo)致兩層之間的耦合,而且我們也不希望其他對象可以直接使用Action對象。
要解決這個問題,我們可以采用的一種方式是,專門在Action中定義一個屬性,用來封裝信息。然后只需要將這個對象傳個service即可。
但是這樣又帶來了一個新問題,誰來將屬性封裝進(jìn)對象呢?答案就是ModelDriven攔截器
攔截器:
攔截器的作用和過濾器類似。
攔截器可以在請求到達(dá)Action之前進(jìn)行攔截,希望在請求到達(dá)Action之前通過攔截器做一些準(zhǔn)備工作。
Struts2 簡單的運(yùn)行流程:
根據(jù)配置信息創(chuàng)建ActionProxy代理類。
判斷:如果ActionMapping為null,不是Struts請求,直接放行
如果ActionMapping不為null,是Struts請求,繼續(xù)處理
請求首先到達(dá)StrutsPrepareAndExecuteFilter.doFilter()
在doFilter方法中,先獲取ActionMapping
通過configurationManager加載Struts的配置信息,找到請求對應(yīng)的Action對象
在StrutsActionProxy.execute()方法中調(diào)用DefaultActionInvocation.invoke()方法
對所有的攔截器進(jìn)行迭代在去分別調(diào)用攔截器intercept方法,進(jìn)行攔截請求
intercept方法我們對請求進(jìn)行一些處理,處理完畢以后繼續(xù)DefaultActionInvocation.invoke()方法
如此反復(fù)直到所有的攔截器都被調(diào)用
最后才去執(zhí)行Action的方法。
2. 請求參數(shù)在哪封裝的:
請求參數(shù)在到達(dá)Action之前,會先經(jīng)過ParametersInterceptor攔截器,
在該攔截器中會自動對將請求參數(shù)封裝進(jìn)值棧的棧頂對象,