這個是不久前在面試的時候遇到的一個問題,當時直接懵了,兩個單拎出來,雖然不太徹底,但都大概知道能夠對請求進行攔截,放在一塊兒比較,可真是頭疼。java
其實以前面試完就去學習了一波,只不過那個時候沒及時總結,如今總結一下,以避免往後遇到這類問題又給忘咯。web
要理解這類問題,光靠死記硬背可能當時有用,過一陣子就差很少忘了。要想真的牢記,咱們必需要實操一下。面試
首先,要使用Filter,必須實現javax.servlet.Filter
接口:spring
public interface Filter { //web應用加載進容器,Filter對象建立以後,執行init方法初始化,用於加載資源,只執行一次。 public default void init(FilterConfig filterConfig) throws ServletException {} //每次請求或響應被攔截時執行,可執行屢次。 public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException; //web應用移除容器,服務器被正常關閉,則執行destroy方法,用於釋放資源,只執行一次。 public default void destroy() {} }
FilterChain
對象用來調用下一個過濾器。public interface HandlerInterceptor { //攔截handler的執行 --> 在HanlerMapping決定適合的handler以後,[在HandlerAdater調用handler以前執行。] default boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { return true; } //攔截handler的執行 --> [在HandlerAdapter調用handler以後],在DispatcherServlet渲染視圖以前執行 default void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception { } //視圖渲染後調用,且只有preHandle結果爲true,纔會調用 default void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception { } }
//DispatcherServlet if (!mappedHandler.applyPreHandle(processedRequest, response)) { return; //遍歷全部的interceptors,調用preHandle方法,只有返回true,才能進行下去 } // 這裏也就是處理Contrller mv = ha.handle(processedRequest, response, mappedHandler.getHandler()); //視圖渲染 applyDefaultViewName(processedRequest, mv); //視圖渲染以後調用 mappedHandler.applyPostHandle(processedRequest, response, mv);
javax.servlet.Filter
接口,Filter使用須要依賴於Tomcat等容器。org.springframework.web.servlet
包下,由Spring容器管理【又有更加豐富的生繆那個週期處理方法,細粒度,且可以使用Spring中的資源】,不依賴Tomcat等容器。這一段在HandlerInterceptor
類的註釋上能夠發現,二者的觸發時機是不一樣的:segmentfault
同時配置了過濾器和攔截器的情形:安全
MyFilter1 前 MyFilter2 前 MyInterceptor1 在Controller前執行 MyInterceptor2 在Controller前執行 controller方法執行... MyInterceptor2 Controller以後,視圖渲染以前 MyInterceptor1 Controller以後,視圖渲染以前 MyInterceptor2 視圖渲染完成以後執行 MyInterceptor1 視圖渲染完成以後執行 MyFilter2 後 MyFilter1 後
每一次都將chain對象傳入,達到最後接口回調的效果:服務器
preHandle1 -> preHande2 -> 【Controller】 -> postHandle2 -> postHandle1 -> afterCompletion2 -> afterComplention1
preHandle按照註冊順序,後兩個與註冊順序相反。app
preHandle
爲false,則以後的全部攔截器都不會執行。preHandle
爲true,則這個攔截器的triggerAfterCompletion
必定會執行。preHandler
都爲true,也就是正常執行,postHandle
纔會執行。boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception { HandlerInterceptor[] interceptors = getInterceptors(); if (!ObjectUtils.isEmpty(interceptors)) { for (int i = 0; i < interceptors.length; i++) { HandlerInterceptor interceptor = interceptors[i]; //一旦當前攔截器preHandle的返回值爲false,那麼從上一個可用的攔截器的afterCompletion開始 if (!interceptor.preHandle(request, response, this.handler)) { triggerAfterCompletion(request, response, null); return false; //這裏返回false意爲 後續不進行下去了。 } this.interceptorIndex = i;//interceptorIndex初始化爲-1,只有當前攔截器preHandle爲true,纔會賦值當前的i。 } } return true; } void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception { HandlerInterceptor[] interceptors = getInterceptors(); if (!ObjectUtils.isEmpty(interceptors)) { for (int i = interceptors.length - 1; i >= 0; i--) { HandlerInterceptor interceptor = interceptors[i]; interceptor.postHandle(request, response, this.handler, mv); } } } void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex) throws Exception { HandlerInterceptor[] interceptors = getInterceptors(); for (int i = this.interceptorIndex; i >= 0; i--) }
二者默認都是使用註冊順序,若是想要認爲控制執行的順序,方式略有不一樣:框架
@Order(2) @Component public class MyFilter1 implements Filter {}
@Component public class WebAdapter implements WebMvcConfigurer { @Autowired MyInterceptor1 myInterceptor1; @Autowired MyInterceptor2 myInterceptor2; @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(myInterceptor1).addPathPatterns("/**").order(2); registry.addInterceptor(myInterceptor2).addPathPatterns("/**").order(1); } }
原理實現上:過濾器基於回調實現,而攔截器基於動態代理。ide
控制粒度上:過濾器和攔截器都可以實現對請求的攔截功能,可是在攔截的粒度上有較大的差別,攔截器對訪問控制的粒度更細。
使用場景上:攔截器每每用於權限檢查、日誌記錄等,過濾器主要用於過濾請求中無效參數,安全校驗。
依賴容器上:過濾器依賴於Servlet容器,侷限於web,而攔截器依賴於Spring框架,可以使用Spring框架的資源,不只限於web。
觸發時機上:過濾器在Servlet先後執行,攔截器在handler先後執行,如今大多數web應用基於Spring,攔截器更細。
參考連接: