SpringMVC的攔截器和過(guò)濾器

轉(zhuǎn):https://blog.csdn.net/xiaoyaotan_111/article/details/53817918

SpringMVC的攔截器(Interceptor)和過(guò)濾器(Filter)的區(qū)別與聯(lián)系

一 簡(jiǎn)介

(1)過(guò)濾器:

依賴(lài)于servlet容器。在實(shí)現(xiàn)上基于函數(shù)回調(diào),可以對(duì)幾乎所有請(qǐng)求進(jìn)行過(guò)濾费奸,但是缺點(diǎn)是一個(gè)過(guò)濾器實(shí)例只能在容器初始化時(shí)調(diào)用一次鲸郊。使用過(guò)濾器的目的是用來(lái)做一些過(guò)濾操作,獲取我們想要獲取的數(shù)據(jù)货邓,比如:在過(guò)濾器中修改字符編碼秆撮;在過(guò)濾器中修改HttpServletRequest的一些參數(shù),包括:過(guò)濾低俗文字换况、危險(xiǎn)字符等

關(guān)于過(guò)濾器的一些用法可以參考我寫(xiě)過(guò)的這些文章

(2)攔截器:

依賴(lài)于web框架职辨,在SpringMVC中就是依賴(lài)于SpringMVC框架。在實(shí)現(xiàn)上基于Java的反射機(jī)制戈二,屬于面向切面編程(AOP)的一種運(yùn)用舒裤。由于攔截器是基于web框架的調(diào)用,因此可以使用Spring的依賴(lài)注入(DI)進(jìn)行一些業(yè)務(wù)操作觉吭,同時(shí)一個(gè)攔截器實(shí)例在一個(gè)controller生命周期之內(nèi)可以多次調(diào)用腾供。但是缺點(diǎn)是只能對(duì)controller請(qǐng)求進(jìn)行攔截,對(duì)其他的一些比如直接訪問(wèn)靜態(tài)資源的請(qǐng)求則沒(méi)辦法進(jìn)行攔截處理

關(guān)于過(guò)濾器的一些用法可以參考我寫(xiě)過(guò)的這些文章:

二 多個(gè)過(guò)濾器與攔截器的代碼執(zhí)行順序

如果在一個(gè)項(xiàng)目中僅僅只有一個(gè)攔截器或者過(guò)濾器鲜滩,那么我相信相對(duì)來(lái)說(shuō)理解起來(lái)是比較容易的伴鳖。但是我們是否思考過(guò):如果一個(gè)項(xiàng)目中有多個(gè)攔截器或者過(guò)濾器,那么它們的執(zhí)行順序應(yīng)該是什么樣的徙硅?或者再?gòu)?fù)雜點(diǎn)榜聂,一個(gè)項(xiàng)目中既有多個(gè)攔截器,又有多個(gè)過(guò)濾器嗓蘑,這時(shí)它們的執(zhí)行順序又是什么樣的呢须肆?

下面我將用簡(jiǎn)單的代碼來(lái)測(cè)試說(shuō)明:

(1)先定義兩個(gè)過(guò)濾器:

i)過(guò)濾器1:

public class TestFilter1 extends OncePerRequestFilter {

protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
        throws ServletException, IOException {
    //在DispatcherServlet之前執(zhí)行
    system.out.println("############TestFilter1 doFilterInternal executed############");
    filterChain.doFilter(request, response);
    //在視圖頁(yè)面返回給客戶端之前執(zhí)行,但是執(zhí)行順序在Interceptor之后
    System.out.println("############TestFilter1 doFilter after############");

// try {
// Thread.sleep(10000);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
}

}

ii)過(guò)濾器2:

public class TestFilter2 extends OncePerRequestFilter {

protected void doFilterInternal(HttpServletRequest request, HttpServletResponse response, FilterChain filterChain)
        throws ServletException, IOException {
    System.out.println("############TestFilter2 doFilterInternal executed############");
    filterChain.doFilter(request, response);
    System.out.println("############TestFilter2 doFilter after############");

}

}

iii)在web.xml中注冊(cè)這兩個(gè)過(guò)濾器:

<!-- 自定義過(guò)濾器:testFilter1 --> 
<filter>
    <filter-name>testFilter1</filter-name>
    <filter-class>cn.zifangsky.filter.TestFilter1</filter-class>
</filter>
<filter-mapping>
    <filter-name>testFilter1</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>
<!-- 自定義過(guò)濾器:testFilter2 --> 
<filter>
    <filter-name>testFilter2</filter-name>
    <filter-class>cn.zifangsky.filter.TestFilter2</filter-class>
</filter>
<filter-mapping>
    <filter-name>testFilter2</filter-name>
    <url-pattern>/*</url-pattern>
</filter-mapping>

(2)再定義兩個(gè)攔截器:

i)攔截器1桩皿,基本攔截器:

public class BaseInterceptor implements HandlerInterceptor{

/**
 * 在DispatcherServlet之前執(zhí)行
 * */
public boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2) throws Exception {
    System.out.println("************BaseInterceptor preHandle executed**********");
    return true;
}

/**
 * 在controller執(zhí)行之后的DispatcherServlet之后執(zhí)行
 * */
public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)
        throws Exception {
    System.out.println("************BaseInterceptor postHandle executed**********");
}

/**
 * 在頁(yè)面渲染完成返回給客戶端之前執(zhí)行
 * */
public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)
        throws Exception {
    System.out.println("************BaseInterceptor afterCompletion executed**********");

// Thread.sleep(10000);
}

}

ii)指定controller請(qǐng)求的攔截器:

public class TestInterceptor implements HandlerInterceptor {

public boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2) throws Exception {
    System.out.println("************TestInterceptor preHandle executed**********");
    return true;
}

public void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)
        throws Exception {
    System.out.println("************TestInterceptor postHandle executed**********");
}

public void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)
        throws Exception {
    System.out.println("************TestInterceptor afterCompletion executed**********");
}

}

iii)在SpringMVC的配置文件中注冊(cè)這兩個(gè)攔截器:

<!-- 攔截器 -->
<mvc:interceptors>
    <!-- 對(duì)所有請(qǐng)求都攔截豌汇,公共攔截器可以有多個(gè) -->
    <bean name="baseInterceptor" class="cn.zifangsky.interceptor.BaseInterceptor" />
    <!-- <bean name="testInterceptor" class="cn.zifangsky.interceptor.TestInterceptor" /> -->
    <mvc:interceptor>       
        <!-- 對(duì)/test.html進(jìn)行攔截 -->
        <mvc:mapping path="/test.html"/>
        <!-- 特定請(qǐng)求的攔截器只能有一個(gè) -->
        <bean class="cn.zifangsky.interceptor.TestInterceptor" />
    </mvc:interceptor>
</mvc:interceptors>

(3)定義一個(gè)測(cè)試使用的controller:

@Controller
public class TestController {

@RequestMapping("/test.html")
public ModelAndView handleRequest(){
    System.out.println("---------TestController executed--------");
    return new ModelAndView("test");
}

}

(4)視圖頁(yè)面test.jsp:

<%@ page language="java" contentType="text/html; charset=UTF-8"
pageEncoding="UTF-8"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<base >
<title>FilterDemo</title>
</head>
<body>
<%
System.out.println("test.jsp is loading");
%>
<div align="center">
This is test page
</div>
</body>
</html>

(5)測(cè)試效果:

啟動(dòng)此測(cè)試項(xiàng)目,可以看到控制臺(tái)中輸出如下:

image.png

這就說(shuō)明了過(guò)濾器的運(yùn)行是依賴(lài)于servlet容器的泄隔,跟springmvc等框架并沒(méi)有關(guān)系拒贱。并且,多個(gè)過(guò)濾器的執(zhí)行順序跟xml文件中定義的先后關(guān)系有關(guān)

接著清空控制臺(tái)中的輸出內(nèi)容并訪問(wèn):http://localhost:9180/FilterDemo/test.html

可以看到梅尤,此時(shí)的控制臺(tái)輸出結(jié)果如下:

image.png

相信從這個(gè)打印輸出柜思,大家就可以很清晰地看到有多個(gè)攔截器和過(guò)濾器存在時(shí)的整個(gè)執(zhí)行順序了岩调。當(dāng)然巷燥,對(duì)于過(guò)個(gè)攔截器它們之間的執(zhí)行順序跟在SpringMVC的配置文件中定義的先后順序有關(guān)

注:對(duì)于整個(gè)SpringMVC的執(zhí)行流程來(lái)說(shuō),如果加上上面的攔截器和過(guò)濾器号枕,其最終的執(zhí)行流程就如下圖所示:

image.png
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末缰揪,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌钝腺,老刑警劉巖抛姑,帶你破解...
    沈念sama閱讀 218,451評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異艳狐,居然都是意外死亡定硝,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,172評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門(mén)毫目,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蔬啡,“玉大人,你說(shuō)我怎么就攤上這事镀虐∠潴。” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 164,782評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵刮便,是天一觀的道長(zhǎng)空猜。 經(jīng)常有香客問(wèn)我,道長(zhǎng)恨旱,這世上最難降的妖魔是什么辈毯? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,709評(píng)論 1 294
  • 正文 為了忘掉前任,我火速辦了婚禮搜贤,結(jié)果婚禮上漓摩,老公的妹妹穿的比我還像新娘。我一直安慰自己入客,他們只是感情好管毙,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,733評(píng)論 6 392
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著桌硫,像睡著了一般夭咬。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上铆隘,一...
    開(kāi)封第一講書(shū)人閱讀 51,578評(píng)論 1 305
  • 那天卓舵,我揣著相機(jī)與錄音,去河邊找鬼膀钠。 笑死掏湾,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的肿嘲。 我是一名探鬼主播融击,決...
    沈念sama閱讀 40,320評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼雳窟!你這毒婦竟也來(lái)了尊浪?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 39,241評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎拇涤,沒(méi)想到半個(gè)月后捣作,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,686評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡鹅士,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,878評(píng)論 3 336
  • 正文 我和宋清朗相戀三年券躁,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片掉盅。...
    茶點(diǎn)故事閱讀 39,992評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡嘱朽,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出怔接,到底是詐尸還是另有隱情搪泳,我是刑警寧澤,帶...
    沈念sama閱讀 35,715評(píng)論 5 346
  • 正文 年R本政府宣布扼脐,位于F島的核電站岸军,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏瓦侮。R本人自食惡果不足惜艰赞,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,336評(píng)論 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望肚吏。 院中可真熱鬧方妖,春花似錦、人聲如沸罚攀。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,912評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)斋泄。三九已至杯瞻,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間炫掐,已是汗流浹背魁莉。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 33,040評(píng)論 1 270
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留募胃,地道東北人旗唁。 一個(gè)月前我還...
    沈念sama閱讀 48,173評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像痹束,于是被迫代替她去往敵國(guó)和親检疫。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,947評(píng)論 2 355

推薦閱讀更多精彩內(nèi)容