拦截器(Interceptor)是一种动态拦截方法调用的机制,在SpringMVC中动态拦截控制器方法的执行
作用:
核心原理:利用AOP思想
首先他们的归属不同:Filter属于Servlet技术,Interceptor属于SpringMVC技术
其次拦截内容不同:Filter对所有访问进行增强,Interceptor仅针对SpringMVC的访问进行增强
请求到达的路线顺序图如下所示:

记住一点就好:过滤器是过滤Servlet的,拦截器是拦截SpringMVC的

Spring MVC中,所有的拦截器都需要实现HandlerInterceptor接口,该接口包含如下三个方法:preHandle()、postHandle()、afterCompletion()。
【第一步】定义拦截器
做法:定义一个类,实现HandlerInterceptor接口即可
注意当前类必须受Spring容器控制,使用@Component告诉spring进行容器管理
@Component //注意当前类必须受Spring容器控制
//定义拦截器类,实现HandlerInterceptor接口
public class ProjectInterceptor implements HandlerInterceptor {
@Override
//原始方法调用前执行的内容
//返回值类型可以拦截控制的执行,true放行,false终止
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle..."+contentType);
return true;
}
@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...");
}
}
【第二步】配置加载拦截器
@Configuration
public class SpringMvcSupport extends WebMvcConfigurationSupport {
@Autowired
private ProjectInterceptor projectInterceptor;
@Override
protected void addInterceptors(InterceptorRegistry registry) {
//配置拦截器
registry.addInterceptor(projectInterceptor)
.addPathPatterns("/users","/users/*");
}
}
上面【第二步】步骤也可以使用标准接口WebMvcConfigurer简化开发(注意:侵入式较强),代码如下所示:
@Configuration
@ComponentScan({"com.yyl.controller"})
@EnableWebMvc
//实现WebMvcConfigurer接口可以简化开发,但具有一定的侵入性
public class SpringMvcConfig implements WebMvcConfigurer {
@Autowired
private ProjectInterceptor projectInterceptor;
@Override
public void addInterceptors(InterceptorRegistry registry) {
//配置多拦截器
registry.addInterceptor(projectInterceptor)
.addPathPatterns("/users","/users/*");
}
}
方法执行流程如下:

我们将有无拦截器的流程进行对比:
我们可以发现拦截器很强大,总的来说是可以用来增强处理器功能的。

在原始方法调用前执行的内容
//原始方法调用前执行的内容
//返回值类型可以拦截控制的执行,true放行,false终止
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle..."+contentType);
return true;
}
参数
| 参数 | 说明 |
|---|---|
| request | 请求对象 |
| response | 响应对象 |
| handler | 被调用的处理器对象,本质上是一个方法对象,对反射技术中的Method对象进行了再包装 |
返回值
在原始方法调用后执行的内容
//原始方法调用后执行的内容
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle...");
}
参数
| 参数 | 说明 |
|---|---|
| request | 请求对象 |
| response | 响应对象 |
| handler | 被调用的处理器对象,本质上是一个方法对象,对反射技术中的Method对象进行了再包装 |
| modelAndView | 如果处理器执行完成具有返回结果,可以读取到对应数据与页面信息,并进行跳转 |
注意:如果处理器方法出现异常了,该方法不会执行
在原始方法调用完成后执行的内容
//原始方法调用完成后执行的内容
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion...");
}
参数
| 参数 | 说明 |
|---|---|
| request | 请求对象 |
| response | 响应对象 |
| handler | 被调用的处理器对象,本质上是一个方法对象,对反射技术中的Method对象进行了再包装 |
| ex | 如果处理器执行过程中出现异常对象,可以针对异常情况进行单独处理 |
注意:无论处理器方法内部是否出现异常,该方法都会执行。
@Component
public class ProjectInterceptor2 implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
System.out.println("preHandle...222");
return false;
}
@Override
public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
System.out.println("postHandle...222");
}
@Override
public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
System.out.println("afterCompletion...222");
}
}
@Configuration
@ComponentScan({"com.itheima.controller"})
@EnableWebMvc
//实现WebMvcConfigurer接口可以简化开发,但具有一定的侵入性
public class SpringMvcConfig implements WebMvcConfigurer {
@Autowired
private ProjectInterceptor projectInterceptor;
@Autowired
private ProjectInterceptor2 projectInterceptor2;
@Override
public void addInterceptors(InterceptorRegistry registry) {
//配置多拦截器
registry.addInterceptor(projectInterceptor)
.addPathPatterns("/books","/books/*");
registry.addInterceptor(projectInterceptor2)
.addPathPatterns("/books","/books/*");
}
}
