1、SpringMVC 中的Interceptor 拦截请求是通过HandlerInterceptor 来实现的。在SpringMVC 中定义一个Interceptor 非常简单,主要有两种方式,第一种方式是要定义的Interceptor类要实现了Spring 的HandlerInterceptor 接口,或者是这个类继承实现了HandlerInterceptor 接口的类,比如Spring 已经提供的实现了HandlerInterceptor 接口的抽象类HandlerInterceptorAdapter ;第二种方式是实现Spring的WebRequestInterceptor接口,或者是继承实现了WebRequestInterceptor的类。
2、继承抽象类HandlerInterceptorAdapter
package com.kp.spring.interceptor;
import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import org.springframework.web.bind.annotation.ResponseBody;
import org.springframework.web.method.HandlerMethod;
import org.springframework.web.servlet.ModelAndView;
import org.springframework.web.servlet.handler.HandlerInterceptorAdapter;
import com.kp.spring.util.SystemUtils;
/**
* @author: py
* @version:2016年10月17日 上午11:28:43
* com.kp.spring.interceptor.TestInterceptor.java
* @Desc SpringMVC 中的Interceptor 拦截器也是相当重要和相当有用的,它的主要作用是拦截用户的请求
* 并进行相应的处理。比如通过它来进行权限验证,或者是来判断用户是否登陆,或者是像12306 那样子判断当前时间是
* 否是购票时间。
*/
/**
* 输入:
* http://localhost:8080/SpringMvcKP/test/test.do
* 结果:
时间:2016-10-22 16:32:15;类:com.kp.spring.interceptor.TestInterceptor;path:/test/test.do
时间:2016-10-22 16:32:15;类:com.kp.spring.interceptor.TestInterceptor;contextPath:/SpringMvcKP
时间:2016-10-22 16:32:15;类:com.kp.spring.interceptor.TestInterceptor;name:com.kp.spring.controller
*
*/
public class TestInterceptor extends HandlerInterceptorAdapter {
/**
* preHandle方法是进行处理器拦截用的,顾名思义,该方法将在Controller处理之前进行调用,
* SpringMVC中的Interceptor拦截器是链式的,可以同时存在 多个Interceptor,
* 然后SpringMVC会根据声明的前后顺序一个接一个的执行,而且所有的Interceptor中的preHandle方法都会在
* Controller方法调用之前调用。
* SpringMVC的这种Interceptor链式结构也是可以进行中断的,这种中断方式是令preHandle的返
* 回值为false,
* 当preHandle的返回值为false的时候整个请求就结束了。
*/
@Override
public boolean preHandle(HttpServletRequest request,
HttpServletResponse response, Object handler) throws Exception {
SystemUtils.println(TestInterceptor.class, null, "****1preHandle");
String path = request.getServletPath();
String contextPath = request.getContextPath() ;
SystemUtils.println(TestInterceptor.class, "path", path);
SystemUtils.println(TestInterceptor.class, "contextPath", contextPath);
if (handler instanceof HandlerMethod) {
HandlerMethod handlerMethod = (HandlerMethod) handler;
if(handlerMethod.getMethodAnnotation(ResponseBody.class)!=null){
//ajax 请求不拦截
return true;
}
// 访问的不是WX下的,直接跳过
String name = handlerMethod.getBeanType().getPackage().getName();
SystemUtils.println(TestInterceptor.class, "name", name);
if (!name.contains("com.oo.project")) {
return true;
}
}
return true;
}
/**
* 这个方法只会在当前这个Interceptor的preHandle方法返回值为true的时候才会执行。
*
* postHandle是进行处理器拦截用的,它的执行时间是在处理器进行处理之后,也就是在Controller
* 的方法调用之后执行,但是它会在DispatcherServlet进行视图的渲染之前执行,也就是说在这个方法中
* 你可以对ModelAndView进行操作。
* 这个方法的链式结构跟正常访问的方向是相反的,也就是说先声明的Interceptor拦截器该方法反而会后调用,
* 这跟Struts2里面的拦截器的执行过程有点像, 只是Struts2里面的intercept方法中要手动的调用
* ActionInvocation的invoke方法,Struts2中调用ActionInvocation的invoke方法就是调用下
* 一个Interceptor或者是调用action,然后要在Interceptor之前调用的内容都写在调用invoke之前,
* 要在Interceptor之后调用的内容都写在调用invoke方法之后。
*/
@Override
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,
ModelAndView modelAndView) throws Exception {
SystemUtils.println(TestInterceptor.class, null, "****2postHandle");
super.postHandle(request, response, handler, modelAndView);
}
/**
* 该方法也是需要当前对应的Interceptor的preHandle方法的返回值为true时才会执行。
* 该方法将在整个请求完成之后,也就是DispatcherServlet渲染了视图执行,
* 这个方法的主要作用是用于清理资源的,
* 当然这个方法也只能在当前这个Interceptor的preHandle方法的返回值为true时才会执行。
*/
@Override
public void afterCompletion(HttpServletRequest request,
HttpServletResponse response, Object handler, Exception ex)
throws Exception {
SystemUtils.println(TestInterceptor.class, null, "****3afterCompletion");
super.afterCompletion(request, response, handler, ex);
}
}
3、两个需拦截的controller
①MyController
package com.kp.spring.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
public class MyController {
@RequestMapping("my.do")
public void my(){
System.out.println("test my ");
}
}
package com.kp.spring.controller;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
@Controller
@RequestMapping("you")
public class YouController {
@RequestMapping("you.do")
public void you(){
System.out.println("test you ");
}
}
4、配置文件如果是
控制台输出:
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****1preHandle
test my
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****2postHandle
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****3afterCompletion
浏览器输入:http://localhost:8080/SpringMvcKP/you/you.do
控制台输出:test you
结果:
成功拦截http://localhost:8080/SpringMvcKP/my.do
拦截 http://localhost:8080/SpringMvcKP/you/you.do失败5、配置文件如果是
浏览器输入:http://localhost:8080/SpringMvcKP/my.do
控制台输出:
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****1preHandle
test my
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****2postHandle
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****3afterCompletion
浏览器输入:http://localhost:8080/SpringMvcKP/you/you.do
控制台输出:
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****1preHandle
test you
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****2postHandle
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****3afterCompletion
结果:
成功拦截http://localhost:8080/SpringMvcKP/you/you.do
成功拦截http://localhost:8080/SpringMvcKP/my.do
6、配置文件如果是,
浏览器输入:http://localhost:8080/SpringMvcKP/my.do
控制台输出:test my
浏览器输入:http://localhost:8080/SpringMvcKP/you/you.do
控制台输出:
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****1preHandle
test you
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****2postHandle
时间:2016-10-22 17:10:46;类:com.kp.spring.interceptor.TestInterceptor;内容:****3afterCompletion
结果:
拦截http://localhost:8080/SpringMvcKP/my.do失败
成功拦截http://localhost:8080/SpringMvcKP/you/you.do
7、配置文件如果是,
浏览器输入:http://localhost:8080/SpringMvcKP/my.do
控制台输出:test my
浏览器输入:http://localhost:8080/SpringMvcKP/you/you.do
控制台输出:test you
结果:
拦截http://localhost:8080/SpringMvcKP/my.do失败
拦截http://localhost:8080/SpringMvcKP/you/you.do失败