您好,登錄后才能下訂單哦!
1. 認(rèn)識(shí)攔截器
Spring MVC的攔截器(Interceptor)不是Filter,同樣可以實(shí)現(xiàn)請(qǐng)求的預(yù)處理、后處理。使用攔截器僅需要兩個(gè)步驟:
1.1 實(shí)現(xiàn)攔截器
實(shí)現(xiàn)攔截器可以自定義實(shí)現(xiàn)HandlerInterceptor接口,也可以通過(guò)繼承HandlerInterceptorAdapter類(lèi),后者是前者的實(shí)現(xiàn)類(lèi)。下面是攔截器的一個(gè)實(shí)現(xiàn)的例子,目的是判斷用戶是否登錄。如果preHandle方法return true,則繼續(xù)后續(xù)處理。
public class LoginInterceptor extends HandlerInterceptorAdapter { /** *預(yù)處理回調(diào)方法,實(shí)現(xiàn)處理器的預(yù)處理(如登錄檢查)。 *第三個(gè)參數(shù)為響應(yīng)的處理器,即controller。 *返回true,表示繼續(xù)流程,調(diào)用下一個(gè)攔截器或者處理器。 *返回false,表示流程中斷,通過(guò)response產(chǎn)生響應(yīng)。 */ @Override public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception { System.out.println("-------------------preHandle"); // 驗(yàn)證用戶是否登陸 Object obj = request.getSession().getAttribute("username"); if (obj == null || !(obj instanceof String)) { response.sendRedirect(request.getContextPath() + "/index.html"); return false; } return true; } /** *當(dāng)前請(qǐng)求進(jìn)行處理之后,也就是Controller 方法調(diào)用之后執(zhí)行, *但是它會(huì)在DispatcherServlet 進(jìn)行視圖返回渲染之前被調(diào)用。 *此時(shí)我們可以通過(guò)modelAndView對(duì)模型數(shù)據(jù)進(jìn)行處理或?qū)σ晥D進(jìn)行處理。 */ @Override public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception { System.out.println("-------------------postHandle"); } /** *方法將在整個(gè)請(qǐng)求結(jié)束之后,也就是在DispatcherServlet 渲染了對(duì)應(yīng)的視圖之后執(zhí)行。 *這個(gè)方法的主要作用是用于進(jìn)行資源清理工作的。 */ @Override public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception { System.out.println("-------------------afterCompletion"); } }
1.2 注冊(cè)攔截器
為了使自定義的攔截器生效,需要注冊(cè)攔截器到spring容器中,具體的做法是繼承WebMvcConfigurerAdapter類(lèi),覆蓋其addInterceptors(InterceptorRegistry registry)方法。最后別忘了把Bean注冊(cè)到Spring容器中,可以選擇@Component 或者 @Configuration。
@Component public class InterceptorConfiguration extends WebMvcConfigurerAdapter{ @Override public void addInterceptors(InterceptorRegistry registry) { // 注冊(cè)攔截器 InterceptorRegistration ir = registry.addInterceptor(new LoginInterceptor()); // 配置攔截的路徑 ir.addPathPatterns("/**"); // 配置不攔截的路徑 ir.excludePathPatterns("/**.html"); // 還可以在這里注冊(cè)其它的攔截器 //registry.addInterceptor(new OtherInterceptor()).addPathPatterns("/**"); } }
1.3 攔截器的應(yīng)用場(chǎng)景
攔截器本質(zhì)上是面向切面編程(AOP),符合橫切關(guān)注點(diǎn)的功能都可以放在攔截器中來(lái)實(shí)現(xiàn),主要的應(yīng)用場(chǎng)景包括:
2. 原理
2.1 工作原理
攔截器不是Filter,卻實(shí)現(xiàn)了Filter的功能,其原理在于:
2.2 攔截器工作流程
一個(gè)攔截器,只有preHandle方法返回true,postHandle、afterCompletion才有可能被執(zhí)行;如果preHandle方法返回false,則該攔截器的postHandle、afterCompletion必然不會(huì)被執(zhí)行。
假設(shè)我們有兩個(gè)攔截器,例如叫Interceptor1和Interceptor2,當(dāng)一個(gè)請(qǐng)求過(guò)來(lái),正常的流程和中斷的流程分別如下。
2.2.1正常流程
注意兩個(gè)攔截器在執(zhí)行preHandle方法和執(zhí)行postHandle、afterCompletion方法時(shí),順序是顛倒的。
2.2.2 中斷流程
假設(shè)執(zhí)行Interceptor2.preHandle中報(bào)錯(cuò),那么流程被中斷,之前被執(zhí)行過(guò)的攔截器的afterCompletion仍然會(huì)執(zhí)行。在本例中,即執(zhí)行了Interceptor1.afterCompletion。
1. Interceptor1.preHandle 2. Interceptor2.preHandle //中間流程被中斷,不再執(zhí)行 3. Interceptor1.afterCompletion
2.3 和Filter共存時(shí)的執(zhí)行順序
攔截器是在DispatcherServlet這個(gè)servlet中執(zhí)行的,因此所有的請(qǐng)求最先進(jìn)入Filter,最后離開(kāi)Filter。其順序如下。
Filter->Interceptor.preHandle->Handler->Interceptor.postHandle->Interceptor.afterCompletion->Filter
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持億速云。
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。