SprimgMVC学习笔记(十)—— 拦截器
一、?什么是攔截器?
Spring MVC中的攔截器(Interceptor)類似于Servlet中的過濾器(Filter),它主要用于攔截用戶請(qǐng)求并作相應(yīng)的處理。例如通過攔截器可以進(jìn)行權(quán)限驗(yàn)證、記錄請(qǐng)求信息的日志、判斷用戶是否登錄等。
要使用Spring MVC中的攔截器,就需要對(duì)攔截器類進(jìn)行定義和配置。通常攔截器可以通過實(shí)現(xiàn)HandlerInterceptor接口,或繼承HandlerInterceptor接口的實(shí)現(xiàn)類(如HandlerInterceptorAdapter)來定義。
public class HandlerInterceptor1 implements HandlerInterceptor {// controller執(zhí)行后且視圖返回后調(diào)用此方法// 這里可得到執(zhí)行controller時(shí)的異常信息// 這里可記錄操作日志 @Overridepublic void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)throws Exception {System.out.println("HandlerInterceptor1....afterCompletion");}// controller執(zhí)行后但未返回視圖前調(diào)用此方法// 這里可在返回用戶前對(duì)模型數(shù)據(jù)進(jìn)行加工處理,比如這里加入公用信息以便頁面顯示 @Overridepublic void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)throws Exception {System.out.println("HandlerInterceptor1....postHandle");}// Controller執(zhí)行前調(diào)用此方法// 返回true表示繼續(xù)執(zhí)行,返回false中止執(zhí)行// 這里可以加入登錄校驗(yàn)、權(quán)限攔截等 @Overridepublic boolean preHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2) throws Exception {System.out.println("HandlerInterceptor1....preHandle");// 設(shè)置為true,測(cè)試使用return true;} }?二、攔截器配置
上面定義的攔截器再?gòu)?fù)制一份HandlerInterceptor2,注意新的攔截器修改代碼:
System.out.println("HandlerInterceptor2....preHandle");在springmvc.xml中配置攔截器:
<mvc:interceptors><mvc:interceptor><!-- 所有請(qǐng)求都進(jìn)入攔截器 --><mvc:mapping path="/**"/><!-- 配置具體的攔截器 --><bean class="cn.itcast.ssm.interceptor.HandlerInterceptor1"></bean></mvc:interceptor><mvc:interceptor><!-- 所有請(qǐng)求都進(jìn)入攔截器 --><mvc:mapping path="/**"/><!-- 配置具體的攔截器 --><bean class="cn.itcast.ssm.interceptor.HandlerInterceptor2"></bean></mvc:interceptor></mvc:interceptors>?三、測(cè)試
3.1 正常流程測(cè)試
瀏覽器訪問地址:http://127.0.0.1:8080/springmvc-web/itemList.action
控制臺(tái)打印:
3.2 中斷流程測(cè)試
瀏覽器訪問地址:http://127.0.0.1:8080/springmvc-web/itemList.action
HandlerInterceptor1的preHandler方法返回false,HandlerInterceptor2返回true:
HandlerInterceptor1的preHandler方法返回true,HandlerInterceptor2返回false:
3.3 結(jié)果分析
從日志看出第一個(gè)攔截器的preHandler方法返回false后第一個(gè)攔截器只執(zhí)行了preHandler方法,其它兩個(gè)方法沒有執(zhí)行,第二個(gè)攔截器的所有方法不執(zhí)行,且Controller也不執(zhí)行了。
第二個(gè)攔截器的preHandler方法返回false后第一個(gè)攔截器的postHandler沒有執(zhí)行,第二個(gè)攔截器的postHandler和afterCompletion沒有執(zhí)行,且controller也不執(zhí)行了。
總結(jié):
preHandle按攔截器定義順序調(diào)用
postHandler按攔截器定義逆序調(diào)用
afterCompletion按攔截器定義逆序調(diào)用
postHandler在攔截器鏈內(nèi)所有攔截器返成功調(diào)用
afterCompletion只有preHandle返回true才調(diào)用
四、攔截器應(yīng)用
4.1 處理流程
1、有一個(gè)登錄頁面,需要寫一個(gè)Controller訪問登錄頁面
2、登錄頁面有一提交表單的動(dòng)作。需要在Controller中處理。
a) 判斷用戶名密碼是否正確(在控制臺(tái)打印)
b) 如果正確,向session中寫入用戶信息(寫入用戶名username)
c) 跳轉(zhuǎn)到商品列表
3、攔截器
a) 攔截用戶請(qǐng)求,判斷用戶是否登錄(登錄請(qǐng)求不能攔截)
b) 如果用戶已經(jīng)登錄。放行
c) 如果用戶未登錄,跳轉(zhuǎn)到登錄頁面。
4.2 編寫登錄jsp
<%@ page language="java" contentType="text/html; charset=UTF-8"pageEncoding="UTF-8"%> <!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd"> <html> <head> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Insert title here</title> </head> <body><form action="${pageContext.request.contextPath }/user/login.action"> <label>用戶名:</label> <br> <input type="text" name="username"> <br> <label>密碼:</label> <br> <input type="password" name="password"> <br> <input type="submit"></form></body> </html>4.3 用戶登錄Controller
@Controller @RequestMapping("user") public class UserController {/*** 跳轉(zhuǎn)到登錄頁面* @return*/@RequestMapping("toLogin")public String toLogin(){return "login";}@RequestMapping("login")public String login(String username,String password,HttpSession session){// 校驗(yàn)用戶登錄 System.out.println(username);System.out.println(password);// 把用戶放到session中session.setAttribute("username", username);return "redirect:/item/itemList.action";} }4.4 編寫攔截器
public class LoginHandlerInterceptor implements HandlerInterceptor {@Overridepublic boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object obj) throws Exception {// 從request中獲取sessionHttpSession session = request.getSession();// 從session中獲取usernameObject username = session.getAttribute("username");// 判斷username是否為nullif(username != null){// 如果不為空,則放行return true;}else{// 如果為空則跳轉(zhuǎn)到登錄頁面response.sendRedirect(request.getContextPath() + "/user/toLogin.action");}return false;}@Overridepublic void afterCompletion(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, Exception arg3)throws Exception {}@Overridepublic void postHandle(HttpServletRequest arg0, HttpServletResponse arg1, Object arg2, ModelAndView arg3)throws Exception {}}4.5 配置攔截器
只能攔截商品的url,所以需要修改ItemController,讓所有的請(qǐng)求都必須以item開頭,如下圖:
在springmvc.xml配置攔截器:
<mvc:interceptor><!-- 配置商品被攔截器攔截 --><mvc:mapping path="/item/**" /><!-- 配置具體的攔截器 --><bean class="cn.itcast.ssm.interceptor.LoginHandlerInterceptor" /> </mvc:interceptor>轉(zhuǎn)載于:https://www.cnblogs.com/yft-javaNotes/p/10216867.html
總結(jié)
以上是生活随笔為你收集整理的SprimgMVC学习笔记(十)—— 拦截器的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: .NET 中使用 Mutex 进行跨越进
- 下一篇: 总结day11 ----函数的学习(2)