自定義攔截器
1.什么是攔截器
說明:
攔截器與過濾器的區別
SpringMVC 的攔截器(Interceptor)與 Java Servlet 的過濾器(Filter)類似,它主要用于攔截用戶的請求并做相應的處理,通常應用在權限驗證、記錄請求資訊的日志、判斷用戶是否登錄等功能上,
-
SpringMVC 也可以使用攔截器對請求進行攔截處理,用戶可以自定義攔截器來實作特定的功能
-
自定義的攔截器必須實作 HandlerInterceptor 介面
-
自定義攔截器的三個方法:
(1)preHandle():該方法在業務處理器處理請求之前被呼叫,在該方法中對用戶請求 request 進行處理
(2)postHandle():該方法在目標方法處理完請求之后執行
(3)afterCompletion():該方法在完全處理完請求之后被呼叫,可以在該方法中進行一些資源清理的操作
2.自定義攔截器執行流程分析圖
● 自定義攔截器執行流程說明
攔截器中方法的執行順序是 preHandle -> Controller -> postHandle -> afterCompletion,只有preHandle回傳true,才會執行后面的方法
- 如果 preHandle 方法回傳 false, 則不再執行目標方法及之后的攔截方法,可以在該方法指定回傳頁面
- postHandle 在目標方法被執行后執行,可以在 postHandle 方法中訪問到目標方法回傳的 ModelAndView 物件
- 若 preHandle 回傳 true, 則 afterCompletion 方法在渲染視圖之后被執行
- 若 preHandle 回傳 false, 則 afterCompletion 方法不會被呼叫
- 配置攔截器時可以指定該攔截器對哪些請求生效,哪些請求不生效,如果不指定則默認對所有目標方法生效
3.應用實體
3.1快速入門
完成一個自定義攔截器,學習如何配置攔截器和攔截器的運行流程,
(1)創建攔截器 MyInterceptor01.java
package com.li.web.interceptor;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* @author 李
* @version 1.0
*/
//需要標識為component注入到spring容器中
@Component
public class MyInterceptor01 implements HandlerInterceptor {
/**
* 1.preHandle 方法在目標方法執行前被執行
* 2.如果 preHandle 方法回傳了false,則目標方法不再被執行
* 3.preHandle方法可以獲取到 request,response,handler
* 4.如果該方法回傳了 false,你可以指定跳轉到哪個頁面
* @param request
* @param response
* @param handler 要執行的處理器
* @return
* @throws Exception
*/
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
Object handler) throws Exception {
System.out.println("===MyInterceptor01--preHandle()===");
return true;
}
/**
* 1.在目標方法執行后,會執行 postHandle 方法
* 2.postHandle 方法可以獲取到目標方法回傳的 modelAndView
* @param request
* @param response
* @param handler
* @param modelAndView
* @throws Exception
*/
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response,
Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("===MyInterceptor01--postHandle()===");
}
/**
* 1.afterCompletion 方法在視圖渲染后被執行
* 2.在 afterCompletion 方法中可以進行一些資源清理作業
* @param request
* @param response
* @param handler
* @param ex
* @throws Exception
*/
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("===MyInterceptor01--afterCompletion()===");
}
}
(2)在 spring 的容器檔案中配置攔截器
<!--配置自定義攔截器-->
<mvc:interceptors>
<!--
1.第一種配置方式
2.直接使用ref參考到對應的 myInterceptor01
3.這種方式會攔截所有的目標方法
-->
<ref bean="myInterceptor01"/>
</mvc:interceptors>
(3)測驗 Controller
package com.li.web.interceptor;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
/**
* @author 李
* @version 1.0
*/
@Controller
public class TestHandler {
@RequestMapping(value = "https://www.cnblogs.com/hi")
public String hi() {
System.out.println("===TestHandler--hi()===");
return "success";
}
@RequestMapping(value = "https://www.cnblogs.com/hello")
public String hello() {
System.out.println("===TestHandler--hello()===");
return "success";
}
}
(4)interceptor.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>測驗自定義攔截器</title>
</head>
<body>
<h1>測驗自定義攔截器</h1>
<a href="https://www.cnblogs.com/liyuelian/p//hi">測驗自定義攔截器-hi</a><br/><br/>
<a href="https://www.cnblogs.com/liyuelian/p//hello">測驗自定義攔截器-hello</a>
</body>
</html>
(5)啟動 tomcat,訪問 interceptor.jsp,分別點擊兩個超鏈接
后臺輸出如下:
可以看到方法的執行順序為:preHandler-->目標方法-->postHandle-->afterCompletion,并且攔截器對Controller 所有的方法都進行了攔截,
3.2注意事項和使用細節
-
默認配置是對所有的目標方法都進行攔截,也可以指定攔截某個目標方法,如在 spring 容器檔案中配置:
<!--配置自定義攔截器--> <mvc:interceptors> <!-- 1.第二種配置方式 2.mvc:mapping path="/hi" 指定要攔截的目標方法的路徑 3.ref bean="myInterceptor01" 指定對哪個攔截器進行配置 --> <mvc:interceptor> <mvc:mapping path="/hi"/> <ref bean="myInterceptor01"/> </mvc:interceptor> </mvc:interceptors> -
mvc:mapping 支持通配符,同時可以指定不對哪些目標方法進行攔截,例如:
<!--配置自定義攔截器--> <mvc:interceptors> <!-- 1.第三種配置方式 2.mvc:mapping path="/h*" 表示攔截 /h 開頭的路徑的目標方法 3.mvc:exclude-mapping path="/hello" 表示不攔截指定路徑的目標方法 4.ref bean="myInterceptor01" 指定對哪個攔截器進行配置 --> <mvc:interceptor> <!--h開頭的目標方法都被攔截--> <mvc:mapping path="/h*"/> <mvc:exclude-mapping path="/hello"/> <ref bean="myInterceptor01"/> </mvc:interceptor> </mvc:interceptors> -
攔截器需要配置才生效,不配置是不生效的
-
如果 preHandle() 方法回傳了 false,就不會執行目標方法(前提是目標方法指定了攔截),你可以在該方法中根據業務指定要跳轉的頁面
3.3Debug執行流程
(1)在自定義攔截器 MyInterceptor01 的 preHandle 方法中打上斷點,點擊 debug
(2)瀏覽器訪問頁面 interceptor.jsp,點擊第一個鏈接,訪問目標方法 hi()
(3)后臺游標跳轉到斷點處,此時 preHandle 方法已經拿到了目標方法,說明 preHandle 方法在目標方法前執行
(4)在目標方法 hi 中添加斷點,點擊 resume,游標跳轉到斷點處
(5)在攔截器的 postHandle 方法中添加斷點,點擊 resume,游標如期跳轉到該斷點處,在該方法中可以拿到目標方法對應的 ModelAndView 物件,其中 view 就是指定的要回傳的頁面,model 就是目標方法的資料,
(6)ModelAndView 物件在 DispatcherServlet 的 resolveViewName 方法進行決議,回傳視圖,視圖在 render 方法中進行視圖渲染
(7)在攔截器的 afterCompletion 方法中打上斷點,點擊 resume,游標跳轉到該斷點處
(8)這一個攔截流程就結束了,再次點擊 resume,就會向客戶端回傳資料
4.多個攔截器
4.1多個攔截器執行流程示意圖

4.2實體演示1-執行流程
4.2.1代碼實作
(1)在3.1中快速入門中已經創建了一個攔截器,現在創建第二個攔截器 MyInterceptor02.java
package com.li.web.interceptor;
import org.springframework.stereotype.Component;
import org.springframework.web.servlet.HandlerInterceptor;
import org.springframework.web.servlet.ModelAndView;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
/**
* @author 李
* @version 1.0
*/
@Component
public class MyInterceptor02 implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("===MyInterceptor02--preHandle()===");
return true;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("===MyInterceptor02--postHandle()===");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("===MyInterceptor02--afterCompletion()===");
}
}
(2)在 spring 的容器檔案中配置攔截器
<!--配置自定義攔截器-->
<mvc:interceptors>
<mvc:interceptor>
<!--h開頭的目標方法都被攔截-->
<mvc:mapping path="/h*"/>
<mvc:exclude-mapping path="/hello"/>
<ref bean="myInterceptor01"/>
</mvc:interceptor>
<!--
第二個攔截器
多個攔截器在執行時,按照配置的順序執行
-->
<mvc:interceptor>
<mvc:mapping path="/h*"/>
<ref bean="myInterceptor02"/>
</mvc:interceptor>
</mvc:interceptors>
(3)interceptor.jsp 不變
(4)完成測驗,瀏覽器訪問 jsp 頁面,點擊訪問目標方法 hi
(5)后端輸出如下:與分析的流程一致,
(6)如果把第二個攔截器的 preHandle 方法回傳值改為 false,重新訪問目標方法測驗,后臺輸出如下:
4.2.2注意事項和使用細節
-
如果第一個攔截器的 preHandle() 回傳 false,后面都不再執行,包括目標方法
-
如果第二個攔截器的 preHandle() 回傳 false,就直接執行第一個攔截器的 afterCompletion() 方法,如果攔截器更多,規則類似,
-
以上兩條規則,都是在目標方法符合被攔截條件的前提下,
4.3實體演示2-跳轉至指定頁面
需求:如果用戶提交的資料有禁用詞(比如:病毒),則在第一個攔截器就回傳,不執行目標方法,跳轉至執行頁面
4.3.1代碼實作
(1)修改 MyInterceptor01.java 的 preHandle 方法
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response,
Object handler) throws Exception {
System.out.println("===MyInterceptor01--preHandle()===");
//獲取用戶提交的關鍵字
String keyword = request.getParameter("keyword");
if ("病毒".equals(keyword)) {
//請求轉發到指定頁面
request.getRequestDispatcher("/WEB-INF/pages/warning.jsp")
.forward(request, response);
return false;
}
System.out.println("keyword=" + keyword);
return true;
}
(2)warning.jsp
<%@ page contentType="text/html;charset=UTF-8" language="java" %>
<html>
<head>
<title>警告</title>
</head>
<body>
<h1>你的資訊中含有違規詞</h1>
</body>
</html>
(3)在 postman 中進行測驗
測驗1:無違禁詞
后臺輸出:
測驗2:含違禁詞
后臺輸出:
5.練習
- 將之前的 SpringMVC 檔案上傳、自定義攔截器相關代碼和案例過一遍
- 簡述 SpringMVC 自定義攔截器作業流程,并畫出示意圖
- debug 自定義攔截器原始碼,加深理,梳理流程
轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/544251.html
標籤:Java
