Springboot項(xiàng)目快速實(shí)現(xiàn)攔截器功能
前言
上一篇文章分享了Springboot項(xiàng)目快速實(shí)現(xiàn)過濾器功能,本篇文章接著來盤一盤攔截器,仔細(xì)研究后會(huì)發(fā)現(xiàn),其實(shí)攔截器和過濾器的功能非常類似,可以理解為面向切面編程的一種具體實(shí)現(xiàn)。下面就其功能特性、工作原理、涉及到的核心類以及具體的實(shí)現(xiàn)方式幾個(gè)方面進(jìn)行梳理,以便在實(shí)際業(yè)務(wù)開發(fā)過程中,可以根據(jù)實(shí)際需要選擇合適的實(shí)現(xiàn)訪求。
環(huán)境配置
jdk版本:1.8
開發(fā)工具:Intellij iDEA 2020.1
springboot:2.3.9.RELEASE
HandleInterceptor介紹
攔截器,在java中只是一種概念,并沒有一個(gè)具體的實(shí)現(xiàn)或標(biāo)準(zhǔn),通常所說的java web的攔截器實(shí)際是指HandleInterceptor接口,這是Spring MVC提供的一套攔截機(jī)制,可以在controller處理請(qǐng)求之后和響應(yīng)處理結(jié)果之后,對(duì)請(qǐng)求信息和響應(yīng)結(jié)果進(jìn)行攔截;但不能修改具體的請(qǐng)求信息和響應(yīng)結(jié)果;Spring MVC的攔截器的概念和servlet的Filter非常類似,但是在執(zhí)行順序上是有所不同的,下面會(huì)重點(diǎn)介紹;從本質(zhì)上來說,SpringMVC的攔截器機(jī)制,是AOP(面向切面編程)的一種具體實(shí)現(xiàn),可以很方面用戶對(duì)實(shí)際業(yè)務(wù)中公共的一些業(yè)務(wù)進(jìn)行橫向抽取,但是和servlet的Filter一樣,有一定的局限性,如能對(duì)請(qǐng)求信息和響應(yīng)結(jié)果進(jìn)行攔截,但不能修改具體的請(qǐng)求信息和響應(yīng)結(jié)果;能攔截controller層的方法,但是不能攔截service層的方法;
工作原理
如果把過濾器和攔截器放在一起來分析其工作原理,就需要再次明確一件事:Filter接口的全限定類名是javax.servlet.Filter,HandleInterceptor接口的全限定類名是org.springframework.web.servlet.HandlerInterceptor,從這里就可以看得出來,F(xiàn)ilter是servlet里就有的接口,HandleInterceptor是Spring中新增的接口,兩個(gè)是來源完全不同的東西,但功能卻很類似,那么放在一起又會(huì)發(fā)生什么奇妙的事呢?
1、過濾器1、過濾器2、攔截1、攔截2對(duì)象,會(huì)在Spring容器啟動(dòng)的過程中,完成bean的注冊(cè);
2、當(dāng)客戶端向服務(wù)端發(fā)起http請(qǐng)求時(shí),在請(qǐng)求到達(dá)具體的controller方法之前,會(huì)先經(jīng)過過濾器、攔截器的處理,其中過濾器的執(zhí)行時(shí)機(jī)要早于攔截器;和過濾器一樣,如果當(dāng)前請(qǐng)求匹配到了多個(gè)攔截器,會(huì)形成一個(gè)攔截器鏈,按照默認(rèn)或指定的優(yōu)先級(jí),依次經(jīng)過各個(gè)攔截器對(duì)象的處理之后,才會(huì)到達(dá)具體的controller方法;
3、到達(dá)具體的controller方法后,開始業(yè)務(wù)處理;得到業(yè)務(wù)處理結(jié)果后,響應(yīng)結(jié)果也會(huì)經(jīng)過請(qǐng)求進(jìn)來時(shí)的所有過濾器、攔截器的處理,不同的是順序與請(qǐng)求進(jìn)來時(shí)完全相反,即先進(jìn)后出;
實(shí)現(xiàn)方式
1、實(shí)現(xiàn)org.springframework.web.servlet.HandlerInterceptor接口;
2、繼承 org.springframework.web.servlet.handler.HandlerInterceptorAdapter類;
核心類
HandlerInterceptor
HandlerInterceptor是SpringMVC提供的實(shí)現(xiàn)攔截器功能的一個(gè)標(biāo)準(zhǔn)接口,接口內(nèi)有三個(gè)方法:
1、preHandler(HttpServletRequest request, HttpServletResponse response, Object handler) ,方法在請(qǐng)求處理之前會(huì)被調(diào)用。該方法在 Interceptor 類中最先執(zhí)行,用來進(jìn)行一些前置初始化操作或是對(duì)當(dāng)前請(qǐng)求做預(yù)處理,也可以進(jìn)行一些判斷來決定請(qǐng)求是否要繼續(xù)進(jìn)行下去。該方法的返回值是 Boolean 類型;當(dāng)它返回 false 時(shí),表示請(qǐng)求結(jié)束,后續(xù)的 Interceptor 和 Controller 都不會(huì)再執(zhí)行;當(dāng)它返回為 true 時(shí),會(huì)繼續(xù)調(diào)用下一個(gè) Interceptor 的 preHandle 方法,如果已經(jīng)是最后一個(gè) Interceptor 的時(shí)候就會(huì)調(diào)用當(dāng)前請(qǐng)求的 Controller 方法;
2、postHandler(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) 方法在當(dāng)前請(qǐng)求處理完成之后,也就是 Controller 方法調(diào)用之后執(zhí)行,但是它會(huì)在 DispatcherServlet 進(jìn)行視圖返回渲染之前被調(diào)用,所以我們可以在這個(gè)方法中對(duì) Controller 處理之后的 ModelAndView 對(duì)象進(jìn)行操作;
3、afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handle, Exception ex) 方法需要在當(dāng)前對(duì)應(yīng)的 Interceptor 類的 preHandle 方法返回值為 true 時(shí)才會(huì)執(zhí)行;該方法將在整個(gè)請(qǐng)求結(jié)束之后,也就是在 DispatcherServlet 渲染了對(duì)應(yīng)的視圖之后執(zhí)行,主要用來進(jìn)行資源清理或釋放。
HandlerInterceptorAdapter
HandlerInterceptorAdapter是一個(gè)抽象類,該抽象類實(shí)現(xiàn)了AsyncHandlerInterceptor接口,而AsyncHandlerInterceptor接口又繼承于HandlerInterceptor,因此可以認(rèn)為HandlerInterceptorAdapter是實(shí)現(xiàn)了HandlerInterceptor接口,但是HandlerInterceptorAdapter是抽象類,實(shí)際上并沒有具體實(shí)現(xiàn),所以在實(shí)現(xiàn)攔截器功能功能的兩種方式本質(zhì)上是一種;
代碼實(shí)現(xiàn)
定義兩個(gè)攔截器:MyInterceptor1和MyInterceptor2,通過一次完成的請(qǐng)求,來分析一下HandlerInterceptor接口的preHandle()、postHandle()、afterCompletion()是如何工作的?
1、定義和注冊(cè)兩個(gè)攔截器:MyInterceptor1和MyInterceptor2,設(shè)置攔截器的攔截路徑為/person/get*,myInterceptor2的優(yōu)先級(jí)高于myInterceptor1;
2、發(fā)起http請(qǐng)求(URL:/person/get);
3、/person/get請(qǐng)求在執(zhí)行PersonController#getPerson()之前,會(huì)先執(zhí)行到MyInterceptor1和MyInterceptor2的preHandler()方法;
4、如果MyInterceptor1和MyInterceptor2的preHandler()方法返回都為true,則會(huì)執(zhí)行到PersonController#getPerson();
5、PersonController#getPerson()執(zhí)行完成后,還沒有返回視圖渲染對(duì)象之前MyInterceptor1和MyInterceptor2的postHandle()方法觸發(fā)執(zhí)行;
6、再然后就是整個(gè)請(qǐng)求處理完之后,MyInterceptor1和MyInterceptor2的afterCompletion()方法觸發(fā)執(zhí)行;
PersonController.java
@Controller @RequestMapping("/person") @Slf4j public class PersonController { @Autowired private IPersonService personService; @GetMapping("/get") @ResponseBody public Person getPerson(Integer id) { Person person = this.personService.get(id); log.info("http://查詢person詳情執(zhí)行完成"); return person; } }
定義WebConfig類實(shí)現(xiàn)WebMvcConfigurer接口,實(shí)現(xiàn)addInterceptors(),完成攔截器的注冊(cè)以及設(shè)置好攔截路徑和優(yōu)先級(jí)順序;
@Configuration public class WebConfig implements WebMvcConfigurer { @Override public void addInterceptors(InterceptorRegistry registry) { registry.addInterceptor(new MyInterceptor2()) .addPathPatterns("/person/get*") .order(1); registry.addInterceptor(new MyInterceptor1()) .addPathPatterns("/person/get*")//設(shè)置攔截請(qǐng)求路徑;*是通配符; .order(2)//設(shè)置攔截器對(duì)象的優(yōu)先級(jí),如果有多個(gè)攔截器對(duì)象,設(shè)置數(shù)字越小,優(yōu)先級(jí)越高; .excludePathPatterns("/test");//設(shè)置排除攔截的請(qǐng)求路徑; } }
定義MyInterceptor1類實(shí)現(xiàn)HandlerInterceptor接口
@Slf4j public class MyInterceptor1 implements HandlerInterceptor { //preHandle方法在請(qǐng)求處理之前被調(diào)用;當(dāng)返回true,則表示可以繼續(xù)后續(xù)請(qǐng)求處理;當(dāng)返回false,則表示請(qǐng)求結(jié)束; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { log.info("http://myInterceptor1的preHandle方法開始執(zhí)行"); log.info("http://請(qǐng)求路徑:{}",request.getRequestURI()); HandlerMethod handlerMethod = (HandlerMethod) handler; log.info("http://攔截類:{},方法:{}",handlerMethod.getBean().getClass().getName(),handlerMethod.getMethod().getName()); Enumeration<String> parameterNames = request.getParameterNames(); while (parameterNames.hasMoreElements()){ String parameterName = parameterNames.nextElement(); String parameterValue = request.getParameter(parameterName); log.info("http://請(qǐng)求參數(shù)>{}:{}",parameterName,parameterValue); } return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { log.info("http://myInterceptor1的postHandle方法開始執(zhí)行"); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { log.info("http://myInterceptor1的afterCompletion方法開始執(zhí)行"); } }
定義MyInterceptor2,繼承HandlerInterceptorAdapter抽象類,重寫HandlerInterceptorAdapter類的方法;
@Slf4j public class MyInterceptor2 extends HandlerInterceptorAdapter { //preHandle方法在請(qǐng)求處理之前被調(diào)用;當(dāng)返回true,則表示可以繼續(xù)后續(xù)請(qǐng)求處理;當(dāng)返回false,則表示請(qǐng)求結(jié)束; @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { log.info("http://myInterceptor1的preHandle方法開始執(zhí)行"); log.info("http://請(qǐng)求路徑:{}",request.getRequestURI()); HandlerMethod handlerMethod = (HandlerMethod) handler; log.info("http://攔截類:{},方法:{}",handlerMethod.getBean().getClass().getName(),handlerMethod.getMethod().getName()); Enumeration<String> parameterNames = request.getParameterNames(); while (parameterNames.hasMoreElements()){ String parameterName = parameterNames.nextElement(); String parameterValue = request.getParameter(parameterName); log.info("http://請(qǐng)求參數(shù)>{}:{}",parameterName,parameterValue); } return true; } @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { log.info("http://myInterceptor1的postHandle方法開始執(zhí)行"); } @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { log.info("http://myInterceptor1的afterCompletion方法開始執(zhí)行"); }
結(jié)果驗(yàn)證
Filter與HandleInterceptor的執(zhí)行順序:在請(qǐng)求處理階段,先經(jīng)過Filter然后再經(jīng)過HandleInterceptor,在響應(yīng)處理階段,先經(jīng)過HandleInterceptor,再經(jīng)過Filter,即先進(jìn)后出;
總結(jié)
攔截器的功能與Filter比較類似,實(shí)現(xiàn)方式也比較簡(jiǎn)單,需要特別注意的是攔截器的執(zhí)行時(shí)機(jī)稍晚于過濾器。那么Spring的AOP與過濾器、攔截器相比,又有哪些不同和需要注意的事項(xiàng)呢?下一篇文章會(huì)把這個(gè)坑給填上。
到此這篇關(guān)于Springboot項(xiàng)目快速實(shí)現(xiàn)攔截器功能的文章就介紹到這了,更多相關(guān)Springboot實(shí)現(xiàn)攔截器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
100-200之間所有素?cái)?shù)求和程序代碼(二個(gè)版本)
寫一個(gè)求100-200之間素?cái)?shù),并求和的程序,大家參考使用吧2013-11-11Java中的SynchronousQueue阻塞隊(duì)列及使用場(chǎng)景解析
這篇文章主要介紹了Java中的SynchronousQueue阻塞隊(duì)列及使用場(chǎng)景解析,SynchronousQueue 是 Java 中的一個(gè)特殊的阻塞隊(duì)列,它的主要特點(diǎn)是它的容量為0,這意味著 SynchronousQueue不會(huì)存儲(chǔ)任何元素,需要的朋友可以參考下2023-12-12判斷以逗號(hào)分隔的字符串中是否包含某個(gè)數(shù)的實(shí)例
下面小編就為大家?guī)硪黄袛嘁远禾?hào)分隔的字符串中是否包含某個(gè)數(shù)的實(shí)例。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-11-11JavaWeb開發(fā)之JSTL標(biāo)簽庫的使用、 自定義EL函數(shù)、自定義標(biāo)簽(帶屬性的、帶標(biāo)簽體的)
這篇文章主要介紹了JavaWeb開發(fā)之JSTL標(biāo)簽庫的使用、 自定義EL函數(shù)、自定義標(biāo)簽(帶屬性的、帶標(biāo)簽體的),需要的朋友可以參考下2017-02-02Java中的MessageFormat.format用法實(shí)例
這篇文章主要介紹了Java中的MessageFormat.format用法實(shí)例,本文先是講解了MessageFormat的語法,然后給出了多個(gè)操作實(shí)例,需要的朋友可以參考下2015-06-06