過濾器和攔截器的區別:前端
①攔截器是基於Java的反射機制的,而過濾器是基於函數回調。
②攔截器不依賴與servlet容器,過濾器依賴與servlet容器。
③攔截器只能對action請求起做用,而過濾器則能夠對幾乎全部的請求起做用。
④攔截器能夠訪問action上下文、值棧裏的對象,而過濾器不能訪問。
⑤在action的生命週期中,攔截器能夠屢次被調用,而過濾器只能在容器初始化時被調用一次。ide
⑥攔截器能夠獲取IOC容器中的各個bean,而過濾器就不行,這點很重要,在攔截器裏注入一個service,能夠調用業務邏輯。函數
寫了點測試代碼,順便整理一下思路,搞清楚這幾者之間的順序:post
1.過濾器是JavaEE標準,採用函數回調的方式進行。是在請求進入容器以後,還未進入Servlet以前進行預處理,而且在請求結束返回給前端這之間進行後期處理。測試
@Override public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain) throws IOException, ServletException { System.out.println("before..."); chain.doFilter(request, response); System.out.println("after..."); }
chain.doFilter(request, response);這個方法的調用做爲分水嶺。事實上調用Servlet的doService()方法是在chain.doFilter(request, response);這個方法中進行的。spa
2.攔截器是被包裹在過濾器之中的。code
@Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("preHandle"); return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("postHandle"); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("afterCompletion"); }
a.preHandle()這個方法是在過濾器的chain.doFilter(request, response)方法的前一步執行,也就是在 [System.out.println("before...")][chain.doFilter(request, response)]之間執行。對象
b.preHandle()方法以後,在return ModelAndView以前進行,能夠操控Controller的ModelAndView內容。生命週期
c.afterCompletion()方法是在過濾器返回給前端前一步執行,也就是在[chain.doFilter(request, response)][System.out.println("after...")]之間執行。圖片
3.SpringMVC的機制是由同一個Servlet來分發請求給不一樣的Controller,其實這一步是在Servlet的service()方法中執行的。因此過濾器、攔截器、service()方法,dispatc()方法的執行順序應該是這樣的,大體畫了個圖:其實很是好測試,本身寫一個過濾器,一個攔截器,而後在這些方法中都加個斷點,一路F8下去就得出告終論。