成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久

您的位置:首頁技術文章
文章詳情頁

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

瀏覽:2日期:2022-08-13 11:45:56
1.過濾器 (Filter)

過濾器的配置比較簡單,直接實現Filter 接口即可,也可以通過@WebFilter注解實現對特定URL攔截,看到Filter 接口中定義了三個方法。

init() :該方法在容器啟動初始化過濾器時被調用,它在 Filter 的整個生命周期只會被調用一次。注意:這個方法必須執行成功,否則過濾器會不起作用。 doFilter() :容器中的每一次請求都會調用該方法, FilterChain 用來調用下一個過濾器 Filter。 destroy(): 當容器銷毀 過濾器實例時調用該方法,一般在方法中銷毀或關閉資源,在過濾器 Filter 的整個生命周期也只會被調用一次

@Componentpublic class MyFilter implements Filter { @Override public void init(FilterConfig filterConfig) throws ServletException {System.out.println('Filter 前置'); } @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {System.out.println('Filter 處理中');filterChain.doFilter(servletRequest, servletResponse); } @Override public void destroy() {System.out.println('Filter 后置'); }}2.攔截器 (Interceptor)

攔截器它是鏈式調用,一個應用中可以同時存在多個攔截器Interceptor, 一個請求也可以觸發多個攔截器 ,而每個攔截器的調用會依據它的聲明順序依次執行。首先編寫一個簡單的攔截器處理類,請求的攔截是通過HandlerInterceptor 來實現,看到HandlerInterceptor 接口中也定義了三個方法。

preHandle() :這個方法將在請求處理之前進行調用。注意:如果該方法的返回值為false ,將視為當前請求結束,不僅自身的攔截器會失效,還會導致其他的攔截器也不再執行。 postHandle():只有在 preHandle() 方法返回值為true 時才會執行。會在Controller 中的方法調用之后,DispatcherServlet 返回渲染視圖之前被調用。 有意思的是:postHandle() 方法被調用的順序跟 preHandle() 是相反的,先聲明的攔截器 preHandle() 方法先執行,而postHandle()方法反而會后執行。 afterCompletion():只有在 preHandle() 方法返回值為true 時才會執行。在整個請求結束之后, DispatcherServlet 渲染了對應的視圖之后執行。

@Componentpublic class MyInterceptor implements HandlerInterceptor { @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {System.out.println('Interceptor 前置');return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {System.out.println('Interceptor 處理中'); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {System.out.println('Interceptor 后置'); }}

將自定義好的攔截器處理類進行注冊,并通過addPathPatterns、excludePathPatterns等屬性設置需要攔截或需要排除的 URL。

@Configurationpublic class MyMvcConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new MyInterceptor()).addPathPatterns('/**');registry.addInterceptor(new MyInterceptor1()).addPathPatterns('/**'); }}3.區別

過濾器 和 攔截器 均體現了AOP的編程思想,都可以實現諸如日志記錄、登錄鑒權等功能,但二者的不同點也是比較多的,接下來一一說明。

3.11、實現原理不同

過濾器和攔截器 底層實現方式大不相同,過濾器 是基于函數回調的,攔截器 則是基于Java的反射機制(動態代理)實現的。這里重點說下過濾器!在我們自定義的過濾器中都會實現一個 doFilter()方法,這個方法有一個FilterChain 參數,而實際上它是一個回調接口。ApplicationFilterChain是它的實現類, 這個實現類內部也有一個 doFilter() 方法就是回調方法。

public interface FilterChain { void doFilter(ServletRequest var1, ServletResponse var2) throws IOException, ServletException;}

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

ApplicationFilterChain里面能拿到我們自定義的xxxFilter類,在其內部回調方法doFilter()里調用各個自定義xxxFilter過濾器,并執行 doFilter() 方法。

public final class ApplicationFilterChain implements FilterChain { @Override public void doFilter(ServletRequest request, ServletResponse response) { ...//省略 internalDoFilter(request,response); } private void internalDoFilter(ServletRequest request, ServletResponse response){ if (pos < n) { //獲取第pos個filterApplicationFilterConfig filterConfig = filters[pos++]; Filter filter = filterConfig.getFilter(); ... filter.doFilter(request, response, this);} } }

而每個xxxFilter 會先執行自身的 doFilter() 過濾邏輯,最后在執行結束前會執行filterChain.doFilter(servletRequest, servletResponse),也就是回調ApplicationFilterChain的doFilter() 方法,以此循環執行實現函數回調。

@Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {filterChain.doFilter(servletRequest, servletResponse); }3.2 使用范圍不同

我們看到過濾器 實現的是 javax.servlet.Filter 接口,而這個接口是在Servlet規范中定義的,也就是說過濾器Filter 的使用要依賴于Tomcat等容器,導致它只能在web程序中使用。

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

而攔截器(Interceptor) 它是一個Spring組件,并由Spring容器管理,并不依賴Tomcat等容器,是可以單獨使用的。不僅能應用在web程序中,也可以用于Application、Swing等程序中。

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

3.3 觸發時機不同

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

過濾器Filter是在請求進入容器后,但在進入servlet之前進行預處理,請求結束是在servlet處理完以后。攔截器 Interceptor 是在請求進入servlet后,在進入Controller之前進行預處理的,Controller 中渲染了對應的視圖之后請求結束。

3.4 攔截的請求范圍不同

在上邊我們已經同時配置了過濾器和攔截器,再建一個Controller接收請求測試一下。

@Controller@RequestMapping()public class Test { @RequestMapping('/test1') @ResponseBody public String test1(String a) {System.out.println('我是controller');return null; }}

項目啟動過程中發現,過濾器的init()方法,隨著容器的啟動進行了初始化。此時瀏覽器發送請求,F12 看到居然有兩個請求,一個是我們自定義的 Controller 請求,另一個是訪問靜態圖標資源的請求。

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

看到控制臺的打印日志如下:

執行順序 :Filter 處理中 -> Interceptor 前置 -> 我是controller -> Interceptor 處理中 -> Interceptor 處理后

Filter 處理中Interceptor 前置Interceptor 處理中Interceptor 后置Filter 處理中

3.5 注入Bean情況不同

在實際的業務場景中,應用到過濾器或攔截器,為處理業務邏輯難免會引入一些service服務。下邊我們分別在過濾器和攔截器中都注入service,看看有什么不同?

@Componentpublic class TestServiceImpl implements TestService { @Override public void a() {System.out.println('我是方法A'); }}

過濾器中注入service,發起請求測試一下 ,日志正常打印出“我是方法A”。

@Autowired private TestService testService; @Override public void doFilter(ServletRequest servletRequest, ServletResponse servletResponse, FilterChain filterChain) throws IOException, ServletException {System.out.println('Filter 處理中');testService.a();filterChain.doFilter(servletRequest, servletResponse); }

Filter 處理中我是方法AInterceptor 前置我是controllerInterceptor 處理中Interceptor 后置

在攔截器中注入service,發起請求測試一下 ,竟然報錯了,debug跟一下發現注入的service怎么是Null啊?

Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用

這是因為加載順序導致的問題,攔截器加載的時間點在springcontext之前,而Bean又是由spring進行管理。

攔截器:老子今天要進洞房;Spring:兄弟別鬧,你媳婦我還沒生出來呢!

解決方案也很簡單,我們在注冊攔截器之前,先將Interceptor 手動進行注入。注意:在registry.addInterceptor()注冊的是getMyInterceptor() 實例。

@Configurationpublic class MyMvcConfig implements WebMvcConfigurer { @Bean public MyInterceptor getMyInterceptor(){System.out.println('注入了MyInterceptor');return new MyInterceptor(); } @Override public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(getMyInterceptor()).addPathPatterns('/**'); }}3.6 控制執行順序不同

實際開發過程中,會出現多個過濾器或攔截器同時存在的情況,不過,有時我們希望某個過濾器或攔截器能優先執行,就涉及到它們的執行順序。

過濾器用@Order注解控制執行順序,通過@Order控制過濾器的級別,值越小級別越高越先執行。

@Order(Ordered.HIGHEST_PRECEDENCE)@Componentpublic class MyFilter2 implements Filter {}

攔截器默認的執行順序,就是它的注冊順序,也可以通過Order手動設置控制,值越小越先執行。

@Override public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new MyInterceptor2()).addPathPatterns('/**').order(2);registry.addInterceptor(new MyInterceptor1()).addPathPatterns('/**').order(1);registry.addInterceptor(new MyInterceptor()).addPathPatterns('/**').order(3); }

看到輸出結果發現,先聲明的攔截器 preHandle() 方法先執行,而postHandle()方法反而會后執行。

postHandle() 方法被調用的順序跟 preHandle() 居然是相反的!如果實際開發中嚴格要求執行順序,那就需要特別注意這一點。

Interceptor1 前置Interceptor2 前置Interceptor 前置我是controllerInterceptor 處理中Interceptor2 處理中Interceptor1 處理中Interceptor 后置Interceptor2 處理后Interceptor1 處理后

那為什么會這樣呢? 得到答案就只能看源碼了,我們要知道controller 中所有的請求都要經過核心組件DispatcherServlet路由,都會執行它的 doDispatch() 方法,而攔截器postHandle()、preHandle()方法便是在其中調用的。

protected void doDispatch(HttpServletRequest request, HttpServletResponse response) throws Exception { try { ........... try { // 獲取可以執行當前Handler的適配器HandlerAdapter ha = getHandlerAdapter(mappedHandler.getHandler());// Process last-modified header, if supported by the handler.String method = request.getMethod();boolean isGet = 'GET'.equals(method);if (isGet || 'HEAD'.equals(method)) { long lastModified = ha.getLastModified(request, mappedHandler.getHandler()); if (logger.isDebugEnabled()) {logger.debug('Last-Modified value for [' + getRequestUri(request) + '] is: ' + lastModified); } if (new ServletWebRequest(request, response).checkNotModified(lastModified) && isGet) {return; }}// 注意: 執行Interceptor中PreHandle()方法if (!mappedHandler.applyPreHandle(processedRequest, response)) { return;}// 注意:執行Handle【包括我們的業務邏輯,當拋出異常時會被Try、catch到】mv = ha.handle(processedRequest, response, mappedHandler.getHandler());if (asyncManager.isConcurrentHandlingStarted()) { return;}applyDefaultViewName(processedRequest, mv);// 注意:執行Interceptor中PostHandle 方法【拋出異常時無法執行】mappedHandler.applyPostHandle(processedRequest, response, mv); }}........... }

看看兩個方法applyPreHandle()、applyPostHandle()具體是如何被調用的,就明白為什么postHandle()、preHandle() 執行順序是相反的了。

boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {HandlerInterceptor[] interceptors = this.getInterceptors();if(!ObjectUtils.isEmpty(interceptors)) { for(int i = 0; i < interceptors.length; this.interceptorIndex = i++) {HandlerInterceptor interceptor = interceptors[i];if(!interceptor.preHandle(request, response, this.handler)) { this.triggerAfterCompletion(request, response, (Exception)null); return false;} }}return true; }

void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception {HandlerInterceptor[] interceptors = this.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); }} }

發現兩個方法中在調用攔截器數組 HandlerInterceptor[] 時,循環的順序竟然是相反的。。。,導致postHandle()、preHandle() 方法執行的順序相反。

到此這篇關于Java中過濾器 (Filter) 和 攔截器 (Interceptor)的使用的文章就介紹到這了,更多相關Java 過濾器和攔截器 內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
成人在线亚洲_国产日韩视频一区二区三区_久久久国产精品_99国内精品久久久久久久
国产精品一区二区三区网站| 欧美成人一区二区三区片免费 | 欧美综合天天夜夜久久| 日本高清不卡aⅴ免费网站| 欧美日韩精品一区二区三区四区| 欧美一级黄色片| 欧美激情一区二区在线| 成人免费小视频| 日韩和欧美一区二区| 国产成人在线看| 伊大人香蕉综合8在线视| 色婷婷综合久久久| 久久综合九色欧美综合狠狠| 一区二区三区精品在线观看| 精品一区二区三区蜜桃| 欧美激情国产日韩| 色婷婷综合视频在线观看| 日韩精品一区二区三区视频| 18欧美乱大交hd1984| 免费观看在线综合| 99热在这里有精品免费| 一本色道久久综合亚洲精品不卡| 91成人在线精品| 久久综合久久综合久久综合| 伊人色综合久久天天| 久久激情五月婷婷| 国产精品大全| 欧美性感一区二区三区| 国产精品午夜免费| 毛片av一区二区| 欧美成人国产| 色呦呦国产精品| 国产欧美日韩在线| 美女精品一区二区| 欧美激情第二页| 欧美性色欧美a在线播放| 国产女主播一区| 久久精品国产99久久6| 欧美.www| 欧美色倩网站大全免费| 国产精品免费视频一区| 韩国av一区二区| 99精品国产99久久久久久福利| 欧美一区二区二区| 亚洲图片欧美一区| www.亚洲在线| 欧美性高清videossexo| 国产精品不卡视频| 国产精品69久久久久水密桃| 亚洲深夜激情| 日本一区二区视频在线| 久久99久久久久| 中日韩视频在线观看| 精品国免费一区二区三区| 天天av天天翘天天综合网色鬼国产| 91老师国产黑色丝袜在线| 欧美影片第一页| 一区二区久久久久久| 97aⅴ精品视频一二三区| 久热国产精品| 亚洲精品欧美综合四区| 午夜精品网站| 精品日韩欧美一区二区| 久久超级碰视频| 久久国产日韩欧美| 一区视频在线播放| 91视频www| 日韩一区二区精品在线观看| 日韩高清欧美激情| 99国产精品久久久久久久| 国产欧美一区在线| 国产福利一区二区三区| 色猫猫国产区一区二在线视频| 日韩一区有码在线| 91浏览器入口在线观看| 日韩欧美中文一区二区| 蜜臀a∨国产成人精品| 国产女优一区| 中文字幕一区二区不卡| 99久久精品国产精品久久| 欧美日韩国产另类不卡| 青青草国产成人av片免费| 亚洲欧美日韩另类精品一区二区三区| 中文字幕 久热精品 视频在线| 99免费精品视频| 欧美一区二区三区成人| 卡一卡二国产精品| 色av综合在线| 免费在线观看一区| 久久精品卡一| 亚洲最新视频在线观看| 日韩视频二区| 亚洲婷婷在线视频| 国产在线成人| 亚洲国产精品高清| 91蝌蚪国产九色| 久久一区二区三区国产精品| yourporn久久国产精品| 精品国产自在久精品国产| 成人国产精品免费| 欧美大白屁股肥臀xxxxxx| 国产成人精品综合在线观看| 欧美一区二区三区白人| 日本欧美一区二区在线观看| 在线区一区二视频| 国产精品v亚洲精品v日韩精品| 99国产欧美久久久精品| 国内国产精品久久| 日产精品久久久久久久性色| 亚洲最新在线观看| 亚洲欧洲日本在线| 99久久99久久精品免费观看| 欧美精品久久99久久在免费线| www.欧美日韩国产在线| 午夜久久久久久久久| 久久美女高清视频 | 色噜噜夜夜夜综合网| 国产精品二区二区三区| 极品销魂美女一区二区三区| 亚洲欧美激情在线| 欧美精品一区二区三区久久久| 一本色道**综合亚洲精品蜜桃冫| 欧美一区精品| 狠狠色狠狠色综合| 亚洲激情图片一区| 日韩一区二区在线观看视频播放 | 欧美精品日韩一区| 国产精品一区二区三区观看| 97精品久久久午夜一区二区三区| 蜜桃视频一区二区三区| 亚洲欧美一区二区三区国产精品 | 美女视频黄 久久| 亚洲精品日韩一| 国产午夜一区二区三区| 欧美三片在线视频观看| 一区二区福利| 91麻豆精品一区二区三区| 黄页网站大全一区二区| 亚洲一二三区在线观看| 国产精品你懂的| 精品国产乱码久久久久久久 | 国产精品成人在线观看| 久久综合九色综合欧美亚洲| 欧美四级电影网| 国产精品毛片在线看| 国产综合网站| 99re视频精品| 大白屁股一区二区视频| 日韩vs国产vs欧美| 亚洲综合一区二区三区| 国产精品无人区| 国产午夜精品美女毛片视频| 日韩一区二区精品葵司在线| 欧美美女一区二区三区| 久久久精品五月天| 99热精品在线观看| 国产综合自拍| 欧美福利电影在线观看| 国产**成人网毛片九色| 激情图片小说一区| 日韩中文字幕91| 亚洲黄色免费网站| 一区免费观看视频| 欧美高清在线一区| 久久久久久麻豆| 日韩精品一区二区三区视频 | 一区二区三区在线看| 国产精品久久久久永久免费观看| 久久免费国产精品| 日韩美女在线视频| 欧美成人综合网站| 2024国产精品| 国产亚洲婷婷免费| 久久九九久久九九| 欧美国产乱子伦| 国产精品热久久久久夜色精品三区| 国产午夜精品久久久久久免费视 | 在线不卡中文字幕| 91麻豆精品国产91久久久久| 欧美日韩视频第一区| 欧美日韩久久不卡| 欧美日韩中文国产| 欧美精品第1页| 欧美一区二区三区的| 日韩午夜激情av| 精品国产一区二区三区久久久蜜月 | 精品视频在线看| 欧美日韩激情一区| 欧美精品xxxxbbbb| 欧美不卡一区二区三区| 久久理论电影网| 国产精品卡一卡二| 日韩美女精品在线| 一区二区三区毛片| 日韩精品一区第一页| 六月丁香婷婷色狠狠久久| 国产在线精品一区二区夜色 | 亚洲激情自拍视频| 天天综合色天天| 久久成人麻豆午夜电影|