面試官:過(guò)濾器和攔截器有什么區(qū)別?

過(guò)濾器(Filter)和攔截器(Interceptor)都是基于 AOP(Aspect Oriented Programming,面向切面編程)思想實(shí)現(xiàn)的,用來(lái)解決項(xiàng)目中某一類問(wèn)題的兩種“工具”,但二者有著明顯的差距,接下來(lái)我們一起來(lái)看。
實(shí)現(xiàn)過(guò)濾器和攔截器
首先,我們先來(lái)看一下二者在 Spring Boot 項(xiàng)目中的具體實(shí)現(xiàn),這對(duì)后續(xù)理解二者的區(qū)別有很大的幫助。
1、實(shí)現(xiàn)過(guò)濾器
過(guò)濾器可以使用 Servlet 3.0 提供的 @WebFilter 注解,配置過(guò)濾的 URL 規(guī)則,然后再實(shí)現(xiàn) Filter 接口,重寫(xiě)接口中的 doFilter 方法,具體實(shí)現(xiàn)代碼如下:
其中:
- void init(FilterConfig filterConfig):容器啟動(dòng)(初始化 Filter)時(shí)會(huì)被調(diào)用,整個(gè)程序運(yùn)行期只會(huì)被調(diào)用一次。用于實(shí)現(xiàn) Filter 對(duì)象的初始化。
- void doFilter(ServletRequest request, ServletResponse response,FilterChain chain):具體的過(guò)濾功能實(shí)現(xiàn)代碼,通過(guò)此方法對(duì)請(qǐng)求進(jìn)行過(guò)濾處理,其中 FilterChain 參數(shù)是用來(lái)調(diào)用下一個(gè)過(guò)濾器或執(zhí)行下一個(gè)流程。
- void destroy():用于 Filter 銷毀前完成相關(guān)資源的回收工作。
2、實(shí)現(xiàn)攔截器
攔截器的實(shí)現(xiàn)分為兩步,第一步,創(chuàng)建一個(gè)普通的攔截器,實(shí)現(xiàn) HandlerInterceptor 接口,并重寫(xiě)接口中的相關(guān)方法;第二步,將上一步創(chuàng)建的攔截器加入到 Spring Boot 的配置文件中。接下來(lái),先創(chuàng)建一個(gè)普通攔截器,實(shí)現(xiàn) HandlerInterceptor 接口并重寫(xiě) preHandle/postHandle/afterCompletion 方法,具體實(shí)現(xiàn)代碼如下:
其中:
- boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handle):在請(qǐng)求方法執(zhí)行前被調(diào)用,也就是調(diào)用目標(biāo)方法之前被調(diào)用。比如我們?cè)诓僮鲾?shù)據(jù)之前先要驗(yàn)證用戶的登錄信息,就可以在此方法中實(shí)現(xiàn),如果驗(yàn)證成功則返回 true,繼續(xù)執(zhí)行數(shù)據(jù)操作業(yè)務(wù);否則就返回 false,后續(xù)操作數(shù)據(jù)的業(yè)務(wù)就不會(huì)被執(zhí)行了。
- void postHandle(HttpServletRequest request, HttpServletResponse response, Object handle, ModelAndView modelAndView):調(diào)用請(qǐng)求方法之后執(zhí)行,但它會(huì)在 DispatcherServlet 進(jìn)行渲染視圖之前被執(zhí)行。
- void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handle, Exception ex):會(huì)在整個(gè)請(qǐng)求結(jié)束之后再執(zhí)行,也就是在 DispatcherServlet 渲染了對(duì)應(yīng)的視圖之后再執(zhí)行。
最后,我們?cè)賹⑸厦娴臄r截器注入到項(xiàng)目配置文件中,并設(shè)置相應(yīng)攔截規(guī)則,具體實(shí)現(xiàn)代碼如下:
了解了二者的使用之后,接下來(lái)我們來(lái)看二者的區(qū)別。
過(guò)濾器 VS 攔截器
過(guò)濾器和攔截器的區(qū)別主要體現(xiàn)在以下 5 點(diǎn):
- 出身不同;
- 觸發(fā)時(shí)機(jī)不同;
- 實(shí)現(xiàn)不同;
- 支持的項(xiàng)目類型不同;
- 使用的場(chǎng)景不同。
接下來(lái),我們一一來(lái)看。
1、出身不同
過(guò)濾器來(lái)自于 Servlet,而攔截器來(lái)自于 Spring 框架,從上面代碼中我們也可以看出,過(guò)濾器在實(shí)現(xiàn)時(shí)導(dǎo)入的是 Servlet 相關(guān)的包,如下圖所示:

而攔截器在實(shí)現(xiàn)時(shí),導(dǎo)入的是 Spring 相關(guān)的包,如下圖所示:

2、觸發(fā)時(shí)機(jī)不同
請(qǐng)求的執(zhí)行順序是:請(qǐng)求進(jìn)入容器 > 進(jìn)入過(guò)濾器 > 進(jìn)入 Servlet > 進(jìn)入攔截器 > 執(zhí)行控制器(Controller),如下圖所示:

所以過(guò)濾器和攔截器的執(zhí)行時(shí)機(jī)也是不同的,過(guò)濾器會(huì)先執(zhí)行,然后才會(huì)執(zhí)行攔截器,最后才會(huì)進(jìn)入真正的要調(diào)用的方法。
3、實(shí)現(xiàn)不同
過(guò)濾器是基于方法回調(diào)實(shí)現(xiàn)的,我們?cè)谏厦鎸?shí)現(xiàn)過(guò)濾器的時(shí)候就會(huì)發(fā)現(xiàn),當(dāng)我們要執(zhí)行下一個(gè)過(guò)濾器或下一個(gè)流程時(shí),需要調(diào)用 FilterChain 對(duì)象的 doFilter 方法進(jìn)行回調(diào)執(zhí)行,如下圖所示:

由此可以看出,過(guò)濾器的實(shí)現(xiàn)是基于方法回調(diào)的。而攔截器是基于動(dòng)態(tài)代理(底層是反射)實(shí)現(xiàn)的,它的實(shí)現(xiàn)如下圖所示:

代理調(diào)用的效果如下圖所示:

4、支持的項(xiàng)目類型不同
過(guò)濾器是 Servlet 規(guī)范中定義的,所以過(guò)濾器要依賴 Servlet 容器,它只能用在 Web 項(xiàng)目中;而攔截器是 Spring 中的一個(gè)組件,因此攔截器既可以用在 Web 項(xiàng)目中,同時(shí)還可以用在 Application 或 Swing 程序中。
5、使用的場(chǎng)景不同
因?yàn)閿r截器更接近業(yè)務(wù)系統(tǒng),所以攔截器主要用來(lái)實(shí)現(xiàn)項(xiàng)目中的業(yè)務(wù)判斷的,比如:登錄判斷、權(quán)限判斷、日志記錄等業(yè)務(wù)。而過(guò)濾器通常是用來(lái)實(shí)現(xiàn)通用功能過(guò)濾的,比如:敏感詞過(guò)濾、字符集編碼設(shè)置、響應(yīng)數(shù)據(jù)壓縮等功能。
本文項(xiàng)目源碼下載
https://gitee.com/mydb/springboot-examples/tree/master/spring-boot-filter
總結(jié)
過(guò)濾器和攔截器都是基于 AOP 思想實(shí)現(xiàn)的,用來(lái)處理某個(gè)統(tǒng)一的功能的,但二者又有 5 點(diǎn)不同:出身不同、觸發(fā)時(shí)機(jī)不同、實(shí)現(xiàn)不同、支持的項(xiàng)目類型不同以及使用的場(chǎng)景不同。過(guò)濾器通常是用來(lái)進(jìn)行全局過(guò)濾的,而攔截器是用來(lái)實(shí)現(xiàn)某項(xiàng)業(yè)務(wù)攔截的。
參考 & 鳴謝
- blog.csdn.net/wo541075754/article/details/111661213
- zhuanlan.zhihu.com/p/340397290

































