JavaWeb中Struts2攔截器深入分析(一)
一、struts2中的攔截器(框架功能核心)
1、過濾器VS攔截器
過濾器VS攔截器功能是一回事。過濾器是Servlet規(guī)范中的技術(shù),可以對請求和響應(yīng)進(jìn)行過濾。
攔截器是Struts2框架中的技術(shù),實(shí)現(xiàn)AOP(面向切面)的編程思想,是可插拔的, 可以對訪問某個(gè) Action 方法之前或之后實(shí)施攔截。
攔截器棧(Interceptor Stack): 將攔截器按一定的順序聯(lián)結(jié)成一條鏈. 在訪問被攔截的方法時(shí), Struts2攔截器鏈中的攔截器就會(huì)按其之前定義的順序被依次調(diào)用
Struts2執(zhí)行原理 - 底層分析
2、自定義攔截器
struts2定義了一個(gè)攔截器接口Interceptor接口。
Interceptor接口里面有三個(gè)抽象方法
•init: 該方法將在攔截器被創(chuàng)建后立即被調(diào)用, 它在攔截器的生命周期內(nèi)只被調(diào)用一次. 可以在該方法中對相關(guān)資源進(jìn)行必要的初始化
•interecept: 每攔截一個(gè)動(dòng)作請求, 該方法就會(huì)被調(diào)用一次.
•destroy: 該方法將在攔截器被銷毀之前被調(diào)用, 它在攔截器的生命周期內(nèi)也只被調(diào)用一次.
Struts 會(huì)依次調(diào)用程序員為某個(gè) Action 而注冊的每一個(gè)攔截器的 interecept 方法.每次調(diào)用 interecept 方法時(shí), Struts 會(huì)傳遞一個(gè) ActionInvocation 接口的實(shí)例.
ActionInvocation: 代表一個(gè)給定動(dòng)作的執(zhí)行狀態(tài), 攔截器可以從該類的對象里獲得與該動(dòng)作相關(guān)聯(lián)的 Action 對象和 Result 對象. 在完成攔截器自己的任務(wù)之后, 攔截器將調(diào)用 ActionInvocation 對象的 invoke 方法前進(jìn)到 Action 處理流程的下一個(gè)環(huán)節(jié).
還可以調(diào)用 ActionInvocation 對象的 addPreResultListener 方法給 ActionInvocation 對象 “掛” 上一個(gè)或多個(gè) PreResultListener 監(jiān)聽器. 該監(jiān)聽器對象可以在動(dòng)作執(zhí)行完畢之后, 開始執(zhí)行動(dòng)作結(jié)果之前做些事情
自定義攔截器步驟:
a、編寫一個(gè)類,實(shí)現(xiàn)com.opensymphony.xwork2.interceptor.Interceptor接口,或者繼承
com.opensymphony.xwork2.interceptor.AbstractInterceptor類。(適配器模式),一般都選擇繼承AbstractInterceptor(攔截器會(huì)駐留內(nèi)存)。因?yàn)锳bstractInterceptor 類實(shí)現(xiàn)了 Interceptor 接口. 并為 init, destroy 提供了一個(gè)空白的實(shí)現(xiàn)
編寫兩個(gè)攔截器InterceptorDemo1 ,和InterceptorDemo2
package com.itheima.interceptor; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.AbstractInterceptor; public class InterceptorDemo1 extends AbstractInterceptor { //動(dòng)作的每次訪問都會(huì)調(diào)用該方法 public String intercept(ActionInvocation invocation) throws Exception { System.out.println("攔截前Demo1"); String rtvalue = invocation.invoke();//放行,這里為什么返回string? 因?yàn)樽罱K的結(jié)果返回的Action的Result,而action的結(jié)果是string類型 System.out.println("攔截后Demo1"); return rtvalue; } } package com.itheima.interceptor; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.AbstractInterceptor; import com.opensymphony.xwork2.interceptor.PreResultListener; public class InterceptorDemo2 extends AbstractInterceptor { //動(dòng)作的每次訪問都會(huì)調(diào)用該方法 public String intercept(ActionInvocation invocation) throws Exception { // invocation.addPreResultListener(new PreResultListener() { // // public void beforeResult(ActionInvocation invocation, String resultCode) { // System.out.println("結(jié)果顯示前"); // } // }); System.out.println("攔截前Demo2"); String rtvalue = invocation.invoke();//放行 System.out.println("攔截后Demo2"); return rtvalue; } }
b、需要在struts.xml中進(jìn)行定義,定義攔截器,先定義在使用。
<package name="p1" extends="struts-default"> <!-- 定義攔截器:只對當(dāng)前包有效 --> <interceptors> <interceptor name="interceprotDemo1" class="com.itheima.interceptor.InterceptorDemo1"></interceptor> <interceptor name="interceprotDemo2" class="com.itheima.interceptor.InterceptorDemo2"></interceptor> </interceptors> </package>
c、在動(dòng)作配置中就可以使用了
<action name="action1" class="com.itheima.action.Demo1Action" method="execute"> <!-- 使用定義的攔截器。如過沒有指定任何的攔截器,默認(rèn)使用default-stack棧中的所有攔截器; 一旦指定了任何一個(gè)攔截器,默認(rèn)的就無效了 --> <interceptor-ref name="interceprotDemo1"></interceptor-ref> <interceptor-ref name="interceprotDemo2"></interceptor-ref> <result>/success.jsp</result> </action>
實(shí)現(xiàn)動(dòng)作類Demo1Action
package com.itheima.action; import com.opensymphony.xwork2.ActionSupport; public class Demo1Action extends ActionSupport { @Override public String execute() throws Exception { System.out.println("execute執(zhí)行了"); return SUCCESS; } }
運(yùn)行結(jié)果
因?yàn)閟truts2中如文件上傳,數(shù)據(jù)驗(yàn)證,封裝請求參數(shù)到action等功能都是由系統(tǒng)默認(rèn)的defaultStack中的攔截器實(shí)現(xiàn)的,所以我們定義的攔截器需要引用系統(tǒng)默認(rèn)的defaultStack,這樣應(yīng)用才可以使用struts2框架提供的眾多功能。
如過沒有指定任何的攔截器,默認(rèn)使用default-stack棧中的所有攔截器;一旦指定了任何一個(gè)攔截器,默認(rèn)的就無效了除了要使用自定義的攔截器之外,還要使用defaultStack,可以這么辦
方法一:(自己使用),只需在action中配置自定義的和defaultStack默認(rèn)的就可以了。
方法二:(大家都用的時(shí)候),如果希望包下的所有action都使用自定義的攔截器, 要使用攔截器棧 interceptor-stack,定義一個(gè)interceptor-stack,然后在action中可以通過<default-interceptor-ref name=“mydefaultStack”/>把攔截器定義為默認(rèn)攔截器,mydefaultStack名字可以自己取。
<interceptors> <interceptor name="interceprotDemo1" class="com.itheima.interceptor.InterceptorDemo1"></interceptor> <interceptor name="interceprotDemo2" class="com.itheima.interceptor.InterceptorDemo2"></interceptor> <interceptor-stack name="mydefaultStack"> <interceptor-ref name="defaultStack"></interceptor-ref> <interceptor-ref name="interceprotDemo1"></interceptor-ref> <interceptor-ref name="interceprotDemo2"></interceptor-ref> </interceptor-stack> </interceptors> <action name="action3" class="com.itheima.action.LoginAction" method="login"> <interceptor-ref name="mydefaultStack"></interceptor-ref> <result>/success.jsp</result> </action>
3、Struts2 自帶的攔截器
案例1:檢查用戶是否登錄
1、 編寫頁面login.jsp
<body> <form action="${pageContext.request.contextPath}/login.action" method="post"> <input type="text" name="username"/><br/> <input type="text" name="password"/><br/> <input type="submit" value="登錄"/> </form> </body>
2、編寫登錄校驗(yàn)的攔截器LoginCheckInterceptor 類
package com.itheima.interceptor; import javax.servlet.http.HttpSession; import org.apache.struts2.ServletActionContext; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.AbstractInterceptor; public class LoginCheckInterceptor extends AbstractInterceptor { public String intercept(ActionInvocation invocation) throws Exception { HttpSession session = ServletActionContext.getRequest().getSession();//通過ServletActionContext對象獲得session對象 Object user = session.getAttribute("user"); if(user==null){ //沒有登錄 return "login";//返回到某個(gè)邏輯視圖 } return invocation.invoke();//放行 } }
3、編寫配置文件struts.xml
<package name="p2" extends="struts-default"> <interceptors> <interceptor name="loginCheckInterceptor" class="com.itheima.interceptor.LoginCheckInterceptor"></interceptor> <interceptor-stack name="mydefaultStack"> <interceptor-ref name="defaultStack"></interceptor-ref> <interceptor-ref name="loginCheckInterceptor"></interceptor-ref> </interceptor-stack> </interceptors> <action name="login" class="com.itheima.action.CustomerAction" method="login"> <result>/login.jsp</result> </action> </package>
4、編寫動(dòng)作類CustomerAction
package com.itheima.action; import org.apache.struts2.ServletActionContext; import com.opensymphony.xwork2.ActionSupport; public class CustomerAction extends ActionSupport { public String login(){ System.out.println("登錄"); ServletActionContext.getRequest().getSession().setAttribute("user", "ppp"); return SUCCESS; } }
案例2:監(jiān)測動(dòng)作方法的執(zhí)行效率
編寫時(shí)間監(jiān)測過濾器TimerInterceptor
package com.itheima.interceptor; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.AbstractInterceptor; public class TimerInterceptor extends AbstractInterceptor { public String intercept(ActionInvocation invocation) throws Exception { long time = System.nanoTime(); String rtvalue = invocation.invoke(); System.out.println(rtvalue+"執(zhí)行耗時(shí):"+(System.nanoTime()-time)+"納秒"); return rtvalue; } }
編寫配置文件
<package name="p2" extends="struts-default"> <interceptors> <interceptor name="loginCheckInterceptor" class="com.itheima.interceptor.LoginCheckInterceptor"></interceptor> <interceptor name="timerInterceptor" class="com.itheima.interceptor.TimerInterceptor"></interceptor> <interceptor-stack name="mydefaultStack"> <interceptor-ref name="defaultStack"></interceptor-ref> <interceptor-ref name="loginCheckInterceptor"></interceptor-ref> <interceptor-ref name="timerInterceptor"></interceptor-ref> </interceptor-stack> </interceptors> <result name="login">/login.jsp</result> </action> </package>
從上面可以看出,在一個(gè)action 中可以配置多個(gè)過濾器。
4、自定義攔截器:能夠指定攔截的方法或不攔截的方法
能夠指定攔截的方法或不攔截的方法,編寫過濾器時(shí),可以實(shí)現(xiàn)類MethodFilterInterceptor,里面有兩個(gè)字段,通過注入?yún)?shù)就可以指定那些不攔截,兩個(gè)參數(shù)只要用一個(gè)即可,當(dāng)攔截較少是,可以用includeMethods ,當(dāng)攔截較多是,可以用排除的方法excludeMethods 。
excludeMethods = Collections.emptySet();//排除那些
includeMethods = Collections.emptySet();//包括那些
案例:再續(xù)登錄校驗(yàn)的例子。
1、編寫過濾器LoginCheckInterceptor
package com.itheima.interceptor; import javax.servlet.http.HttpSession; import org.apache.struts2.ServletActionContext; import com.opensymphony.xwork2.ActionInvocation; import com.opensymphony.xwork2.interceptor.AbstractInterceptor; import com.opensymphony.xwork2.interceptor.MethodFilterInterceptor; public class LoginCheckInterceptor extends MethodFilterInterceptor { protected String doIntercept(ActionInvocation invocation) throws Exception { HttpSession session = ServletActionContext.getRequest().getSession(); Object user = session.getAttribute("user"); if(user==null){ //沒有登錄 return "login";//返回到某個(gè)邏輯視圖 } return invocation.invoke();//放行 } }
2、編寫配置文件
3、編寫動(dòng)作類CustomerAction
package com.itheima.action; import org.apache.struts2.ServletActionContext; import com.opensymphony.xwork2.ActionSupport; public class CustomerAction extends ActionSupport { public String add(){ System.out.println("調(diào)用add的service方法"); return SUCCESS; } public String edit(){ System.out.println("調(diào)用edit的service方法"); return SUCCESS; } public String login(){ System.out.println("登錄"); ServletActionContext.getRequest().getSession().setAttribute("user", "ppp"); return SUCCESS; } }
4、編寫頁面
addCustomer.jsp
<body> 添加客戶 </body>
editCustomer.jsp
<body> 修改客戶 </body>
login.jsp
<body> <form action="${pageContext.request.contextPath}/login.action" method="post"> <input type="text" name="username"/><br/> <input type="text" name="password"/><br/> <input type="submit" value="登錄"/> </form> </body>
success.jsp
<body> oyeah </body>
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
java 讀寫Parquet格式的數(shù)據(jù)的示例代碼
本篇文章主要介紹了java 讀寫Parquet格式的數(shù)據(jù)的示例代碼,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-09-09基于Hutool的圖片驗(yàn)證碼功能模塊實(shí)現(xiàn)
為了提高系統(tǒng)的安全性,防止接口被暴力刷新,驗(yàn)證碼是個(gè)好的手段,圖片驗(yàn)證碼沒有短信驗(yàn)證碼的費(fèi)用,其是個(gè)人開發(fā)者學(xué)習(xí)的重點(diǎn),這篇文章主要介紹了基于Hutool的圖片驗(yàn)證碼功能模塊實(shí)現(xiàn),需要的朋友可以參考下2022-10-10Spring框架應(yīng)用的權(quán)限控制系統(tǒng)詳解
在本篇文章里小編給大家整理的是關(guān)于基于Spring框架應(yīng)用的權(quán)限控制系統(tǒng)的研究和實(shí)現(xiàn),需要的朋友們可以學(xué)習(xí)下。2019-08-08MyBatis-Plus動(dòng)態(tài)表名的使用
本文主要介紹了MyBatis-Plus動(dòng)態(tài)表名的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-04-04如何通過ServletInputStream讀取http請求傳入的數(shù)據(jù)
這篇文章主要介紹了如何通過ServletInputStream讀取http請求傳入的數(shù)據(jù),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-10-10Flink流處理引擎零基礎(chǔ)速通之?dāng)?shù)據(jù)的抽取篇
今天不分享基礎(chǔ)概念知識(shí)了,來分享一個(gè)馬上工作需要的場景,要做數(shù)據(jù)的抽取,不用kettle,想用flink。實(shí)際就是flink的sql、table層級的api2022-05-05