拦截器
作用
是在面向切面编程(体现了AOP思想),可以在controller执行前后进行一些操作
实现
实现HandlerInterceptor接口,或者继承实现了HandlerInterceptor接口的类
HandlerInterceptor
接口中定义了三个方法:
preHandle()
这个方法在业务处理器处理请求之前被调用,SpringMVC中的Interceptor是链式的调用,在一个应用中或者是一个请求中可以存在多个Interceptor。每个Interceptor 的调用会依据它的声明顺序依次执行,而且最先执行的都是Interceptor中的preHandle方法,所以可以在这个方法中进行一些前置初始化操作或者是对当前请求的一个预处理,也可以在这个方法中进行一些判断来决定请求是否要继续进行下去。该方法的返回值是布尔型,当它返回为false时,表示请求结束,后续的Interceptor和Controller都不会执行;;当返回值为true时就会继续调用下一个Interceptor的preHandle方法,如果已经是最后一个Interceptor的时候就会嗲用当前请求的Controlle方法。
postHandle()
这个方法在当前请求处理之后,也就是Controller方法调用之后执行,但是它会在DispatcherServlet进行视图返回渲染之前被调用,所以我们可在这个方法中对Controller处理之后的ModelAndView对象进行操作。postHandle方法被调用的方向跟preHandle是相反的,也就是说先声明的Interceptor 的postHandle 方法反而会后执行。
afterCompletion()
该方法也是需要当前对应的Interceptor的preHandl方法的返回值为true时才会执行。顾名思义,该方法将在整个请求结束之后,也就是在DispatcherServlet渲染了对应的视图之后执行。这个方法的主要作用是用于进行资源清理工作的。
拦截器的调用顺序
代码实例
配置文件
@Configuration
public class WebConfigurer implements WebMvcConfigurer {
@Resource
private Interceptor1 interceptor1;
@Resource
private Interceptor2 interceptor2;
@Override
public void addInterceptors(InterceptorRegistry registry) {
registry.addInterceptor(interceptor1).addPathPatterns("/Usr/**");
registry.addInterceptor(interceptor2).addPathPatterns("/Usr/**");
}
}
拦截器1
@Component
@Slf4j
public class Interceptor1 implements HandlerInterceptor {
@Override
public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
log.info("拦截器1的preHandle...");
return true;
}
@Override
public void postHandle(HttpServletRequest request,
HttpServletResponse response, Object handler,