Spring MVC 源码分析之 DispatcherServlet#getHandlerAdapter 方法

前言:

前面我们分析了 Spring MVC 的工作流程源码,其核心是 DispatcherServlet#doDispatch 方法,我们前面分析了获取 Handler 的方法 DispatcherServlet#getHandler 方法,本篇我们重点分析一下获取当前请求的适配器 HandlerAdapter 的实现原理,具体方法入口是 DispatcherServlet#getHandlerAdapter。

Spring MVC 知识传送门:

详解 Spring MVC(Spring MVC 简介)

Spring MVC 初始化源码分析

Spring MVC 工作流程源码分析

Spring MVC 源码分析之 DispatcherServlet#getHandler 方法

** DispatcherServlet#getHandlerAdapter方法源码分析**

DispatcherServlet#getHandlerAdapter方法就是从 handlerAdapters 中查询匹配当前请求的 Handler,只要找到了就不在循环直接返回,我们我们重点关注adapter.supports(handler) 这行代码,这里实际调用的是接口的抽象类 AbstractHandlerMapping 中的 getHandler 方法,下面接着分析。

java 复制代码
//org.springframework.web.servlet.DispatcherServlet#getHandlerAdapter
protected HandlerAdapter getHandlerAdapter(Object handler) throws ServletException {
	this.handlerAdapters 为空判断 DispatcherServlet 初始化时注册的 handlerAdapters
	if (this.handlerAdapters != null) {
		//迭代遍历
		Iterator var2 = this.handlerAdapters.iterator();

		while(var2.hasNext()) {
			HandlerAdapter adapter = (HandlerAdapter)var2.next();
			//找到匹配当前 handler的 adapter
			if (adapter.supports(handler)) {
				//找到就返回
				return adapter;
			}
		}
	}
	//如果最后都没找到 抛出异常
	throw new ServletException("No adapter for handler [" + handler + "]: The DispatcherServlet configuration needs to include a HandlerAdapter that supports this handler");
}

AbstractHandlerMethodAdapter#supports 方法源码分析

AbstractHandlerMethodAdapter#supports 没有什么复杂的逻辑,只是判断了 handler 是否是 HandlerMethod 类型,至于 AbstractHandlerMethodAdapter#supportsInternal 方法它默认返回 fasle,也就是说只要 handler 是 HandlerMethod 类型,就算匹配成功。

java 复制代码
//org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter#supports
public final boolean supports(Object handler) {
	//handler 是否是 HandlerMethod 类型 
	// this.supportsInternal 抽象方法 由子类实现  RequestMappingHandlerAdapter#supportsInternal 默认返回 true
	return handler instanceof HandlerMethod && this.supportsInternal((HandlerMethod)handler);
}
//org.springframework.web.servlet.mvc.method.AbstractHandlerMethodAdapter#supportsInternal
protected abstract boolean supportsInternal(HandlerMethod var1);

//org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerAdapter#supportsInternal
protected boolean supportsInternal(HandlerMethod handlerMethod) {
	return true;
}

HandlerExecutionChain#applyPreHandle 方法源码分析

HandlerExecutionChain#applyPreHandle 方法的主要左右就是调用拦截器的 preHandle 方法,如果有某个拦截器的 preHandle 方法返回 false,就会逆向调用返回 true 的拦截器的 triggerAfterCompletion 方法。

java 复制代码
//org.springframework.web.servlet.HandlerExecutionChain#applyPreHandle
boolean applyPreHandle(HttpServletRequest request, HttpServletResponse response) throws Exception {
	//获取所有拦截器数组
	HandlerInterceptor[] interceptors = this.getInterceptors();
	//为空判断
	if (!ObjectUtils.isEmpty(interceptors)) {
		//循环调用 拦截器的 preHandle 方法
		//this.interceptorIndex 记录当前拦截器的位置
		for(int i = 0; i < interceptors.length; this.interceptorIndex = i++) {
			HandlerInterceptor interceptor = interceptors[i];
			//判断拦截器的 preHandle 方法返回值
			if (!interceptor.preHandle(request, response, this.handler)) {
				//拦截器 preHandle 方法返回 false 则反向调用返回 true 的那些拦截器的 afterCompletion 方法
				this.triggerAfterCompletion(request, response, (Exception)null);
				//返回 false
				return false;
			}
		}
	}

	return true;
}


//org.springframework.web.servlet.HandlerExecutionChain#triggerAfterCompletion
void triggerAfterCompletion(HttpServletRequest request, HttpServletResponse response, @Nullable Exception ex) throws Exception {
	//获取所有拦截器
	HandlerInterceptor[] interceptors = this.getInterceptors();
	//为空判断
	if (!ObjectUtils.isEmpty(interceptors)) {
		//反向遍历 这里是的 interceptorIndex 上面记录了的 --i 就是反向遍历
		for(int i = this.interceptorIndex; i >= 0; --i) {
			HandlerInterceptor interceptor = interceptors[i];

			try {
				//调用拦截器的 afterCompletion 方法
				interceptor.afterCompletion(request, response, this.handler, ex);
			} catch (Throwable var8) {
				logger.error("HandlerInterceptor.afterCompletion threw exception", var8);
			}
		}
	}

}

HandlerExecutionChain#applyPostHandle 方法源码分析

拦截器的 preHandle 方法调用完成后,就会调用 handle 方法处理具体请求(后面分析),handle 方法调用完成后就会调用拦截器的 applyPreHandle 方法, HandlerExecutionChain#applyPreHandle 方法的主要左右就是逆向调用拦截器的 postHandle 方法。

java 复制代码
//org.springframework.web.servlet.HandlerExecutionChain#applyPostHandle
void applyPostHandle(HttpServletRequest request, HttpServletResponse response, @Nullable ModelAndView mv) throws Exception {
	//获取所有拦截器
	HandlerInterceptor[] interceptors = this.getInterceptors();
	//为空判断
	if (!ObjectUtils.isEmpty(interceptors)) {
		for(int i = interceptors.length - 1; i >= 0; --i) {
			//逆向调用拦截器的 postHandle 方法
			HandlerInterceptor interceptor = interceptors[i];
			interceptor.postHandle(request, response, this.handler, mv);
		}
	}

}

本篇主要分析了 Spring MVC 工作流程中比较简单的几个关键点,HandlerAdapter 的适配过程、拦截器的前置后置处理等,这些在流程中比较简单的环节就放在一起分析了,希望可以帮助到有需要的朋友。

欢迎提出建议及对错误的地方指出纠正。

相关推荐
沙子迷了蜗牛眼9 分钟前
当展示列表使用 URL.createObjectURL 的创建临时图片、视频无法加载问题
java·前端·javascript·vue.js
ganshenml11 分钟前
【Android】 开发四角版本全解析:AS、AGP、Gradle 与 JDK 的配套关系
android·java·开发语言
我命由我1234512 分钟前
Kotlin 运算符 - == 运算符与 === 运算符
android·java·开发语言·java-ee·kotlin·android studio·android-studio
小途软件18 分钟前
ssm327校园二手交易平台的设计与实现+vue
java·人工智能·pytorch·python·深度学习·语言模型
alonewolf_9922 分钟前
Java类加载机制深度解析:从双亲委派到热加载实战
java·开发语言
追梦者12323 分钟前
springboot整合minio
java·spring boot·后端
云游26 分钟前
Jaspersoft Studio community edition 7.0.3的应用
java·报表
帅气的你31 分钟前
Spring Boot 集成 AOP 实现日志记录与接口权限校验
java·spring boot
zhglhy1 小时前
Spring Data Slice使用指南
java·spring
win x1 小时前
Redis 主从复制
java·数据库·redis