成人免费xxxxx在线视频软件_久久精品久久久_亚洲国产精品久久久_天天色天天色_亚洲人成一区_欧美一级欧美三级在线观看

一臉懵逼,面試官:過濾器和攔截器有啥區別?

開發 前端
一臉懵逼,面試官:過濾器和攔截器有啥區別?這個是不久前在面試的時候遇到的一個問題,當時直接懵了,兩個單拎出來,雖然不太完全,但都大概知道可以對請求進行攔截,放在一起比較,可真是頭疼。

這個是不久前在面試的時候遇到的一個問題,當時直接懵了,兩個單拎出來,雖然不太完全,但都大概知道可以對請求進行攔截,放在一起比較,可真是頭疼。

其實之前面試完就去學習了一波,只不過那個時候沒及時總結,現在總結一下,以免日后遇到這類問題又給忘咯。

要理解這類問題,光靠死記硬背可能當時有用,過一陣子就差不多忘了。要想真的牢記,我們必須要實操一下。

Filter的使用

首先,要使用Filter,必須實現javax.servlet.Filter接口:

  1. public interface Filter { 
  2.     //web應用加載進容器,Filter對象創建之后,執行init方法初始化,用于加載資源,只執行一次。  
  3.     public default void init(FilterConfig filterConfig) throws ServletException {} 
  4.     //每次請求或響應被攔截時執行,可執行多次。 
  5.     public void doFilter(ServletRequest request, ServletResponse response, 
  6.             FilterChain chain) throws IOException, ServletException; 
  7.     //web應用移除容器,服務器被正常關閉,則執行destroy方法,用于釋放資源,只執行一次。 
  8.     public default void destroy() {} 
  • init和destroy是default方法,實現類可以不用實現。
  • doFilter必須實現,也就是說,作為一個過濾器,doFilter必須要定義。
  • doFlilter方法中傳進來的FilterChain對象用來調用下一個過濾器。

攔截器的使用

 

  1. public interface HandlerInterceptor { 
  2.     //攔截handler的執行 --> 在HanlerMapping決定適合的handler之后,[在HandlerAdater調用handler之前執行。] 
  3.     default boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) 
  4.             throws Exception {      return true
  5.     }   //攔截handler的執行 --> [在HandlerAdapter調用handler之后],在DispatcherServlet渲染視圖之前執行 
  6.     default void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, 
  7.             @Nullable ModelAndView modelAndView) throws Exception { }   //視圖渲染后調用,且只有preHandle結果為true,才會調用 
  8.     default void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, 
  9.             @Nullable Exception ex) throws Exception {  }} 
  10.  
  11. //DispatcherServlet 
  12. if (!mappedHandler.applyPreHandle(processedRequest, response)) { 
  13.     return; //遍歷所有的interceptors,調用preHandle方法,只有返回true,才能進行下去 
  14. // 這里也就是處理Contrller 
  15. mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); 
  16. //視圖渲染 
  17. applyDefaultViewName(processedRequest, mv); 
  18. //視圖渲染之后調用 
  19. mappedHandler.applyPostHandle(processedRequest, response, mv); 

過濾器與攔截器到底有啥區別呢?

一、實現原理不同

過濾器的實現基于回調函數

攔截器基于Java的反射機制【動態代理】實現。

二、使用范圍不同

過濾器是Servlet的規范,需要實現javax.servlet.Filter接口,Filter使用需要依賴于Tomcat等容器。

攔截器是Spring組件,定義在org.springframework.web.servlet包下,由Spring容器管理【又有更加豐富的生繆那個周期處理方法,細粒度,且能夠使用Spring中的資源】,不依賴Tomcat等容器。

三、觸發時機不同

這一段在HandlerInterceptor類的注釋上可以發現,兩者的觸發時機是不同的:

 

一臉懵逼,面試官:過濾器和攔截器有啥區別?
  • 過濾器:對請求在進入后Servlet之前或之后進行處理。
  • 攔截器:對請求在handler【Controller】前后進行處理。

 

一臉懵逼,面試官:過濾器和攔截器有啥區別?

四、執行順序不同

同時配置了過濾器和攔截器的情形:

  1. MyFilter1 前 
  2. MyFilter2 前 
  3. MyInterceptor1 在Controller前執行 
  4. MyInterceptor2 在Controller前執行 
  5. controller方法執行... 
  6. MyInterceptor2 Controller之后,視圖渲染之前 
  7. MyInterceptor1 Controller之后,視圖渲染之前 
  8. MyInterceptor2 視圖渲染完成之后執行 
  9. MyInterceptor1 視圖渲染完成之后執行 
  10. MyFilter2 后 
  11. MyFilter1 后 

過濾器的順序

每一次都將chain對象傳入,達到最后接口回調的效果:

 

一臉懵逼,面試官:過濾器和攔截器有啥區別?

攔截器的順序

preHandle1 -> preHande2 -> 【Controller】 -> postHandle2 -> postHandle1 -> afterCompletion2 -> afterComplention1preHandle按照注冊順序,后兩個與注冊順序相反。

  • 一個攔截器的preHandle為false,則之后的所有攔截器都不會執行。
  • 一個攔截器的preHandle為true,則這個攔截器的triggerAfterCompletion一定會執行。
  • 只有所有的攔截器preHandler都為true,也就是正常執行,postHandle才會執行。
  1. boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception { 
  2.     HandlerInterceptor[] interceptors = getInterceptors();    if (!ObjectUtils.isEmpty(interceptors)) { 
  3.         for (int i = 0; i < interceptors.length; i++) { 
  4.             HandlerInterceptor interceptor = interceptors[i];            //一旦當前攔截器preHandle的返回值為false,那么從上一個可用的攔截器的afterCompletion開始 
  5.             if (!interceptor.preHandle(request, response, this.handler)) { 
  6.                 triggerAfterCompletion(request, response, null); 
  7.                 return false; //這里返回false意為 后續不進行下去了。 
  8.             }            this.interceptorIndex = i;//interceptorIndex初始化為-1,只有當前攔截器preHandle為true,才會賦值當前的i。 
  9.         }    }    return true
  10. }void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv)    throws Exception {    HandlerInterceptor[] interceptors = getInterceptors();    if (!ObjectUtils.isEmpty(interceptors)) { 
  11.         for (int i = interceptors.length - 1; i >= 0; i--) { 
  12.             HandlerInterceptor interceptor = interceptors[i];            interceptor.postHandle(request, response, this.handler, mv); 
  13.         }    }}void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex)    throws Exception {    HandlerInterceptor[] interceptors = getInterceptors();        for (int i = this.interceptorIndex; i >= 0; i--)  

五、控制執行順序方式不同

兩者默認都是使用注冊順序,如果想要認為控制執行的順序,方式略有不同:

  • 過濾器如果想要強制改變,可以使用@Order注解。
  • 攔截器如果使用order()方法

 

  1. @Order(2) 
  2. @Component 
  3. public class MyFilter1 implements Filter {} 
  4.  
  5. @Component 
  6. public class WebAdapter implements WebMvcConfigurer { 
  7.     @Autowired 
  8.     MyInterceptor1 myInterceptor1;    @Autowired 
  9.     MyInterceptor2 myInterceptor2;    @Override 
  10.     public void addInterceptors(InterceptorRegistry registry) {        registry.addInterceptor(myInterceptor1).addPathPatterns("/**").order(2); 
  11.         registry.addInterceptor(myInterceptor2).addPathPatterns("/**").order(1); 
  12.     }} 

總結

  • 原理實現上:過濾器基于回調實現,而攔截器基于動態代理。
  • 控制粒度上:過濾器和攔截器都能夠實現對請求的攔截功能,但是在攔截的粒度上有較大的差異,攔截器對訪問控制的粒度更細。
  • 使用場景上:攔截器往往用于權限檢查、日志記錄等,過濾器主要用于過濾請求中無效參數,安全校驗。
  • 依賴容器上:過濾器依賴于Servlet容器,局限于web,而攔截器依賴于Spring框架,能夠使用Spring框架的資源,不僅限于web。
  • 觸發時機上:過濾器在Servlet前后執行,攔截器在handler前后執行,現在大多數web應用基于Spring,攔截器更細。

 

責任編輯:未麗燕 來源: 今日頭條
相關推薦

2023-02-20 07:19:14

2021-11-12 06:39:51

Tomcat連接器面試

2024-02-01 08:08:53

Spring過濾器類型Gateway

2022-01-13 10:04:21

攔截器Interceptor過濾器

2023-05-29 07:36:04

Java過濾器攔截器

2024-09-25 17:44:08

2024-01-17 08:56:31

2023-01-26 02:16:17

2020-09-08 06:32:57

項目低耦合高內聚

2020-06-04 07:45:07

過濾器和攔截器

2020-12-11 09:20:01

數據庫架構緩存

2022-09-30 19:32:36

ES面試查詢

2021-05-10 08:01:12

BeanFactoryFactoryBean容器

2022-09-23 18:16:25

KafkaJVM

2021-01-06 10:09:38

MySQL

2023-02-17 08:10:24

2024-01-08 08:33:53

AOPSpring攔截器

2024-04-03 15:33:04

JWTSession傳輸信息

2024-09-19 08:42:43

2025-05-12 18:49:52

點贊
收藏

51CTO技術棧公眾號

主站蜘蛛池模板: 一区二区三区四区国产 | 亚洲国产精品一区二区第一页 | 亚洲第一在线视频 | 精品96久久久久久中文字幕无 | 香蕉久久久| 国产一区二区麻豆 | 做a视频| 久久在线视频 | 天天看天天爽 | 国产激情视频网址 | www.一区二区 | 国产午夜精品一区二区三区四区 | 午夜伦理影院 | 一区二区在线不卡 | 国产精品久久久久久妇女6080 | 国产伦精品一区二区三毛 | 欧美男人天堂 | 日韩中文字幕在线播放 | 午夜精品网站 | 户外露出一区二区三区 | 日日操视频| 国产激情网 | 国产日韩精品一区 | 一区二区高清在线观看 | 亚洲精品一区二区久 | 国产精品日韩欧美一区二区三区 | 色婷婷综合久久久中字幕精品久久 | av一二三区 | 久久久亚洲一区 | 日本在线视频一区二区 | 97色免费视频 | 国产人免费人成免费视频 | 国产精品一区二区三 | 国产午夜亚洲精品不卡 | 国产精品久久久久久久久久久久 | 国产91精品久久久久久久网曝门 | a黄视频 | 亚洲欧美在线观看 | 毛片大全 | 久久草视频 | 一区二区三区国产精品 |