過濾器和攔截器的區(qū)別:
- 攔截器是基于java的反射機(jī)制的蒸其,而過濾器是基于函數(shù)回調(diào)哺窄。
- 攔截器不依賴與servlet容器字柠,過濾器依賴與servlet容器绑榴。
- 攔截器只能對action請求起作用,而過濾器則可以對幾乎所有的請求起作用蹂午。
- 攔截器可以訪問action上下文栏豺、值棧里的對象,而過濾器不能訪問豆胸。
- 在action的生命周期中奥洼,攔截器可以多次被調(diào)用,而過濾器只能在容器初始化時(shí)被調(diào)用一次晚胡。
- 攔截器可以獲取IOC容器中的各個(gè)bean灵奖,而過濾器就不行,這點(diǎn)很重要估盘,在攔截器里注入一個(gè)service瓷患,可以調(diào)用業(yè)務(wù)邏輯。
攔截器可以獲取ioc中的service bean實(shí)現(xiàn)業(yè)務(wù)邏輯忿檩,攔截器可以獲取ioc中的service bean實(shí)現(xiàn)業(yè)務(wù)邏輯尉尾,攔截器可以獲取ioc中的service bean實(shí)現(xiàn)業(yè)務(wù)邏輯,
觸發(fā)時(shí)機(jī):
- 過濾器是在請求進(jìn)入容器后燥透,但請求進(jìn)入servlet之前進(jìn)行預(yù)處理的。請求結(jié)束返回也是辨图,是在servlet處理完后班套,返回給前端之前。
- 過濾器的觸發(fā)時(shí)機(jī)是容器后故河,servlet之前吱韭,所以過濾器的doFilter(
ServletRequest request, ServletResponse response, FilterChain chain
)的入?yún)⑹荢ervletRequest ,而不是httpservletrequest。因?yàn)檫^濾器是在httpservlet之前理盆。 - 過濾器
@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);這個(gè)方法的調(diào)用作為分水嶺痘煤。事實(shí)上調(diào)用Servlet的doService()方法是在chain.doFilter(request, response);這個(gè)方法中進(jìn)行的。
總結(jié):過濾器包裹住servlet猿规,servlet包裹住攔截器衷快。
1.過濾器是JavaEE標(biāo)準(zhǔn),采用函數(shù)回調(diào)的方式進(jìn)行姨俩。是在請求進(jìn)入容器之后蘸拔,還未進(jìn)入Servlet之前進(jìn)行預(yù)處理,并且在請求結(jié)束返回給前端這之間進(jìn)行后期處理环葵。
@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);這個(gè)方法的調(diào)用作為分水嶺调窍。事實(shí)上調(diào)用Servlet的doService()方法是在chain.doFilter(request, response);這個(gè)方法中進(jìn)行的。
2.攔截器是被包裹在過濾器之中的张遭。
@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");
}
preHandle()這個(gè)方法是在過濾器的chain.doFilter(request, response)方法的前一步執(zhí)行邓萨,也就是在 [System.out.println("before...")][chain.doFilter(request, response)]之間執(zhí)行。
preHandle()方法之后菊卷,在return ModelAndView之前進(jìn)行缔恳,可以操控Controller的ModelAndView內(nèi)容。
afterCompletion()方法是在過濾器返回給前端前一步執(zhí)行的烁,也就是在[chain.doFilter(request, response)][System.out.println("after...")]之間執(zhí)行褐耳。
3.SpringMVC的機(jī)制是由同一個(gè)Servlet來分發(fā)請求給不同的Controller,其實(shí)這一步是在Servlet的service()方法中執(zhí)行的渴庆。所以過濾器铃芦、攔截器、service()方法襟雷,dispatc()方法的執(zhí)行順序應(yīng)該是這樣的刃滓,大致畫了個(gè)圖:其實(shí)非常好測試,自己寫一個(gè)過濾器耸弄,一個(gè)攔截器咧虎,然后在這些方法中都加個(gè)斷點(diǎn),一路F8下去就得出了結(jié)論计呈。
4.SpringMVC的機(jī)制是由同一個(gè)Servlet來分發(fā)請求給不同的Controller砰诵,其實(shí)這一步是在Servlet的service()方法中執(zhí)行的。
5.捌显,SpringMVC的機(jī)制是由同一個(gè)Servlet來分發(fā)請求給不同的Controller茁彭,其實(shí)這一步是在Servlet的service()方法中執(zhí)行的。
6.SpringMVC的機(jī)制是由同一個(gè)Servlet來分發(fā)請求給不同的Controller扶歪,其實(shí)這一步是在Servlet的service()方法中執(zhí)行的理肺。
7.還有,攔截器是spring容器的,是spring支持的妹萨,
總結(jié):攔截器功在對請求權(quán)限鑒定方面確實(shí)很有用處年枕,在我所參與的這個(gè)項(xiàng)目之中,第三方的遠(yuǎn)程調(diào)用每個(gè)請求都需要參與鑒定乎完,所以這樣做非常方便熏兄,而且他是很獨(dú)立的邏輯,這樣做讓業(yè)務(wù)邏輯代碼很干凈囱怕。和框架的其他功能一樣霍弹,原理很簡單,使用起來也很簡單娃弓,大致看了下SpringMVC這一部分的源碼典格,其實(shí)還是比較容易理解的。
我們項(xiàng)目中僅僅用到了preHandle這個(gè)方法台丛,而未用其他的耍缴,框架提供了一個(gè)已經(jīng)實(shí)現(xiàn)了攔截器接口的適配器類HandlerInterceptorAdapter,繼承這個(gè)類然后重寫一下需要用到的方法就行了挽霉,可以少幾行代碼防嗡,這種方式Java中很多地方都有體現(xiàn)。