SpringMVC的拦截器

一、拦截器 Interceptor 简介

SpringMVC的拦截器Interceptor规范,主要是对Controller资源访问时进行拦截操作的技术,当然拦截后可以进行权 限控制,功能增强等都是可以的。拦截器有点类似 Javaweb 开发中的Filter,拦截器与Filter的区别如下图:

由上图,对Filter 和 Interceptor 做个对比:

实现了HandlerInterceptor接口,且被Spring管理的Bean都是拦截器,接口定义如下:

java 复制代码
public interface HandlerInterceptor {

    	default boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    		return true;
    	}
    	
    	default void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable ModelAndView modelAndView) throws Exception {
    	
    	}
    	
    	default void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, @Nullable Exception ex) throws Exception {
    	
    	}
    }

HandlerInterceptor接口方法的作用及其参数、返回值详解如下:

二、拦截器快速入门

编写MyInterceptor01实现HandlerInterceptor接口:

java 复制代码
    public class MyInterceptor01 implements HandlerInterceptor {
    	@Override
    	public boolean preHandle(HttpServletRequest request, HttpServletResponse response, Object handler) throws Exception {
    		System.out.println("Controller方法执行之前...");
    		return true;//放行
    	}
    	
    	@Override
    	public void postHandle(HttpServletRequest request, HttpServletResponse response, Object handler, ModelAndView modelAndView) throws Exception {
    		System.out.println("Controller方法执行之后...");
    	}
    	
    	@Override
    	public void afterCompletion(HttpServletRequest request, HttpServletResponse response, Object handler, Exception ex) throws Exception {
    		System.out.println("渲染视图结束,整个流程完毕...");
    	}
    }

配置Interceptor

xml 复制代码
    <!--配置拦截器-->
    <mvc:interceptors>
    	<mvc:interceptor>
    		<!--配置对哪些资源进行拦截操作-->
    		<mvc:mapping path="/**"/>
    		<bean class="com.itheima.interceptor.MyInterceptor01"></bean>
    	</mvc:interceptor>
    </mvc:interceptors>

三、拦截器中三个方法的执行顺序

当每个拦截器都是放行状态时,三个方法的执行顺序如下:

当Interceptor1和Interceptor2处于放行,Interceptor3处于不放行时,三个方法的执行顺序如下:

拦截器执行顺序取决于 interceptor 的配置顺序

xml 复制代码
    <mvc:interceptors>
    	<mvc:interceptor>
    		<mvc:mapping path="/target"/>
    		<bean class="com.itheima.interceptor.MyInterceptor02"></bean>
    		</mvc:interceptor>
    	<mvc:interceptor>
    		<mvc:mapping path="/*"/>
    		<bean class="com.itheima.interceptor.MyInterceptor01"></bean>
    	</mvc:interceptor>
    </mvc:interceptors>

四、拦截器执行原理

请求到来时会先使用组件HandlerMapping去匹配Controller的方法(Handler)和符合拦截路径的Interceptor,Handler和多个Interceptor被封装成一个HandlerExecutionChain的对象
HandlerExecutionChain 定义如下:

java 复制代码
    public class HandlerExecutionChain {
    	// 映射的Controller的方法
    	private final Object handler;
    	// 当前Handler匹配的拦截器集合
    	private final List<HandlerInterceptor> interceptorList;
    	// ... 省略其他代码 ...
    }

在DispatcherServlet的doDispatch方法中执行拦截器

java 复制代码
    protected void doDispatch(HttpServletRequest request, HttpServletResponse response) {
    	// 根据请求信息获得HandlerExecutionChain
    	HandlerExecutionChain mappedHandler = this.getHandler(request);
    	// 获得处理器适配器
    	HandlerAdapter ha = this.getHandlerAdapter(mappedHandler.getHandler());
    	// 执行Interceptor的前置方法,前置方法如果返回false,则该流程结束
    	if (!mappedHandler.applyPreHandle(request, response)) {
    		return;
    	}
    	// 执行handler,一般是HandlerMethod
    	ModelAndView mv = ha.handle(processedRequest, response, mappedHandler.getHandler());
    	// 执行后置方法
    	mappedHandler.applyPostHandle(processedRequest, response, mv);
    	// 执行最终方法
    	this.triggerAfterCompletion(processedRequest, response, mappedHandler, e);
    }

跟踪 HandlerExecutionChain的applyPreHandle方法源码:

java 复制代码
    boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
    	// 对interceptorList进行遍历,正向遍历,与此同时使用interceptorIndex进行计数
    	for(int i = 0; i < this.interceptorList.size(); this.interceptorIndex = i++) {
    		// 取出每一个Interceptor对象
    		HandlerInterceptor interceptor = (HandlerInterceptor)this.interceptorList.get(i);
    		// 调用Interceptor的preHandle方法,如果返回false,则直接执行Interceptor的最终方法
    		if (!interceptor.preHandle(request, response, this.handler)) {
    			// 执行Interceptor的最终方法
    			this.triggerAfterCompletion(request, response, (Exception)null);
    			return false;
    		}
    	}
    	return true;
    }

跟踪 HandlerExecutionChain的applyPostHandle方法源码:

java 复制代码
    void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception {
    	// 对interceptorList进行遍历,逆向遍历
    	for(int i = this.interceptorList.size() - 1; i >= 0; --i) {
    		// 取出每一个Interceptor
    		HandlerInterceptor interceptor = (HandlerInterceptor)this.interceptorList.get(i);
    		// 执行Interceptor的postHandle方法
    		interceptor.postHandle(request, response, this.handler, mv);
    	}
    }

跟踪HandlerExecutionChain的triggerAfterCompletion方法源码:

java 复制代码
    void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex) {
    // 逆向遍历interceptorList,遍历的个数为执行的applyPreHandle次数-1
    for(int i = this.interceptorIndex; i >= 0; --i) {
    	// 取出每一个Interceptor
    	HandlerInterceptor interceptor = (HandlerInterceptor)this.interceptorList.get(i);
    		try {
    			// 执行Interceptor的afterCompletion方法
    			interceptor.afterCompletion(request, response, this.handler, ex);
    		} catch (Throwable var7) {
    			logger.error("HandlerInterceptor.afterCompletion threw exception", var7);
    		}
    	}
    }
相关推荐
程序猿零零漆4 小时前
SpringCloud系列教程:微服务的未来(二十)Seata快速入门、部署TC服务、微服务集成Seata
java·spring·spring cloud·微服务
Miketutu15 小时前
Spring MVC消息转换器
java·spring
小小虫码16 小时前
项目中用的网关Gateway及SpringCloud
spring·spring cloud·gateway
带刺的坐椅21 小时前
无耳科技 Solon v3.0.7 发布(2025农历新年版)
java·spring·mvc·solon·aop
精通HelloWorld!1 天前
使用HttpClient和HttpRequest发送HTTP请求
java·spring boot·网络协议·spring·http
LUCIAZZZ1 天前
基于Docker以KRaft模式快速部署Kafka
java·运维·spring·docker·容器·kafka
拾忆,想起1 天前
如何选择Spring AOP的动态代理?JDK与CGLIB的适用场景
spring boot·后端·spring·spring cloud·微服务
鱼骨不是鱼翅1 天前
Spring Web MVC基础第一篇
前端·spring·mvc
hong_zc1 天前
Spring MVC (三) —— 实战演练
java·spring·mvc
Future_yzx1 天前
Spring AOP 入门教程:基础概念与实现
java·开发语言·spring