攔截器
SpringMVC 中的 Interceptor 攔截器是非常重要和相當有用的,它的主要作用是攔截指定的使用者請求,并進行相應的預處理與後處理。其攔截的時間點在“處理器映射器根據使用者送出的請求映射出了所要執行的處理器類,并且也找到了要執行該處理器類的處理器擴充卡,在處理器擴充卡執行處理器之前”。當然,在處理器映射器映射出所要執行的處理器類時,已經将攔截器與處理器組合為了一個處理器執行鍊,并傳回給了中央排程器。
自定義攔截器實作步驟
![](https://img.laitimes.com/img/9ZDMuAjOiMmIsIjOiQnIsISPrdEZwZ1Rh5WNXp1bwNjW1ZUba9VZwlHdsATOfd3bkFGazxCMx8VesATMfhHLlN3XnxCMwEzX0xiRGZkRGZ0Xy9GbvNGLpZTY1EmMZVDUSFTU4VFRR9Fd4VGdsYTMfVmepNHLrJXYtJXZ0F2dvwVZnFWbp1zczV2YvJHctM3cv1Ce-cmbw5SMxYmNykDMhNTYzczMjRTOmZjMzEWM2cDO3MmM4ITNl9CXzIzLchDMxIDMy8CXn9Gbi9CXzV2Zh1WavwVbvNmLvR3YxUjL0M3Lc9CX6MHc0RHaiojIsJye.png)
攔截器的執行
自定義攔截器,需要實作 HandlerInterceptor 接口。而該接口中含有三個方法:
public class MyInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
return HandlerInterceptor.super.preHandle(request, response, handler);
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
HandlerInterceptor.super.postHandle(request, response, handler, modelAndView);
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
HandlerInterceptor.super.afterCompletion(request, response, handler, ex);
}
}
➢
preHandle(HttpServletRequest request, HttpServletResponse response, Object handler)
:
該方法在處理器方法執行之前執行。其傳回值為 boolean,若為 true,則緊接着會執行處理器方法,且會将 afterCompletion()方法放入到一個專門的方法棧中等待執行。
postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView)
該方法在處理器方法執行之後執行。處理器方法若最終未被執行,則該方法不會執行。由于該方法是在處理器方法執行完後執行,且該方法參數中包含ModelAndView,是以該方法可以修改處理器方法的處理結果資料,且可以修改跳轉方向。
afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex)
當 preHandle()方法傳回 true 時,會将該方法放到專門的方法棧中,等到對請求進行響應的所有工作完成之後才執行該方法。即該方法是在中央排程器渲染(資料填充)了響應頁面之後執行的,此時對 ModelAndView 再操作也對響應無濟于事。
afterCompletion 最後執行的方法,清除資源,例如在 Controller 方法中加入資料。
多個攔截器執行
攔截器登入驗證例子
- 建立controller,doLogin方法不進行攔截,doSome方法進行攔截。doLogin方法進行驗證登入資訊。
@Controller
@RequestMapping("/user")
public class UserController {
@RequestMapping("/login.do")
public ModelAndView doLogin(HttpSession session, String name, Integer age){
ModelAndView modelAndView = new ModelAndView();
modelAndView.addObject("uname",name);
modelAndView.addObject("uage",age);
modelAndView.setViewName("forward:/user/some.do");
if(name.equals("黃振聰")) {
session.setAttribute("uname",name);
}
return modelAndView;
}
@RequestMapping("/some.do")
public String doSome(){
return "target";
}
}
- 建立攔截器實作類。
public class UserInterceptor implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("執行preHandle...");
HttpSession session = request.getSession();
String uname = (String) session.getAttribute("uname");
if(uname != null){
return true;
}
return false;
}
@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...");
}
}
- 在主配置檔案中聲明攔截器
<mvc:interceptors>
<mvc:interceptor>
<mvc:mapping path="/user/some.do"/>
<bean class="com.hzc.interceptor.UserInterceptor" />
</mvc:interceptor>
</mvc:interceptors>
- 建立登入界面與結果界面
<body>
<form action="user/login.do">
姓名:<input type="text" name="name"><br>
年齡:<input type="text" name="age"><br>
<input type="submit" value="登入">
</form>
</body>
<body>
<h2>姓名:${requestScope.uname}</h2>
<h2>年齡:${requestScope.uage}</h2>
</body>