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

前言:

上篇我们分析了 Spring MVC 的工作流程源码,其核心是 DispatcherServlet#doDispatch 方法,这个方法中有获取映射器处理器操作,也就是调用 DispatcherServlet#getHandler 方法,本篇我们重点分析一下 DispatcherServlet#getHandler 的实现原理。

Spring MVC 知识传送门:

详解 Spring MVC(Spring MVC 简介)

Spring MVC 初始化源码分析

Spring MVC 工作流程源码分析

DispatcherServlet#getHandler 方法源码分析

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

//org.springframework.web.servlet.DispatcherServlet#getHandler
@Nullable
protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {//this.handlerMappings 为空判断 DispatcherServlet 初始化时注册的 handlerMappingif (this.handlerMappings != null) {//不为空 迭代遍历Iterator var2 = this.handlerMappings.iterator();while(var2.hasNext()) {HandlerMapping mapping = (HandlerMapping)var2.next();//获取具体的 HandlerExecutionChain 重点关注HandlerExecutionChain handler = mapping.getHandler(request);if (handler != null) {//不为空 返回return handler;}}}return null;
}

AbstractHandlerMapping#getHandler 方法源码分析

AbstractHandlerMapping#getHandler 方法主要作用是获取当前请求的 HandlerExecutionChain,HandlerExecutionChain 包含了 HandlerMapping 和 拦截器,同时也对跨域请求做了一些处理,我们重点关注获取 Handler 和返回拦截器链的部分。

//org.springframework.web.servlet.handler.AbstractHandlerMapping#getHandler
@Nullable
public final HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {//获取调用方法的 handler  重点关注Object handler = this.getHandlerInternal(request);//handler 为空判断if (handler == null) {//为空 获取默认的handler handler = this.getDefaultHandler();}//再次为空判断if (handler == null) {//还为空 直接返回return null;} else {//handler 是否是 String 类型if (handler instanceof String) {//获取 hanlerNameString handlerName = (String)handler;//从容器中获取具体的 handlerhandler = this.obtainApplicationContext().getBean(handlerName);}//获取当前请求的拦截器执行链 重点关注HandlerExecutionChain executionChain = this.getHandlerExecutionChain(handler, request);if (this.logger.isTraceEnabled()) {this.logger.trace("Mapped to " + handler);} else if (this.logger.isDebugEnabled() && !request.getDispatcherType().equals(DispatcherType.ASYNC)) {this.logger.debug("Mapped to " + executionChain.getHandler());}//跨域相关处理if (this.hasCorsConfigurationSource(handler) || CorsUtils.isPreFlightRequest(request)) {CorsConfiguration config = this.corsConfigurationSource != null ? this.corsConfigurationSource.getCorsConfiguration(request) : null;CorsConfiguration handlerConfig = this.getCorsConfiguration(handler, request);config = config != null ? config.combine(handlerConfig) : handlerConfig;executionChain = this.getCorsHandlerExecutionChain(request, executionChain, config);}//返回拦截器链return executionChain;}
}

AbstractHandlerMethodMapping#getHandlerInternal 方法源码分析

getHandlerInternal 方法是由 AbstractHandlerMapping 子类实现的,比如 AbstracUrlHandlerMapping、AbstractHandlerMethodMapping, 这里我们分析 AbstractHandlerMethodMapping#getHandlerInternal 方法,当前方法为了线程安全加了读锁,方法本身没有太多的逻辑,从 Request 中获取到 urlPaht 之后,就继续调用本类的 lookupHandlerMethod 方法,返回 HandlerMethod。

//org.springframework.web.servlet.handler.AbstractHandlerMethodMapping#getHandlerInternal
protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {//从request 中解析出 urlpathString lookupPath = this.initLookupPath(request);//加读锁this.mappingRegistry.acquireReadLock();HandlerMethod var4;try {//根据 urlpath 和 request 寻找具体的 HandlerMethod 重点关注HandlerMethod handlerMethod = this.lookupHandlerMethod(lookupPath, request);var4 = handlerMethod != null ? handlerMethod.createWithResolvedBean() : null;} finally {//释放读锁this.mappingRegistry.releaseReadLock();}return var4;
}

AbstractHandlerMethodMapping#lookupHandlerMethod 方法源码分析

AbstractHandlerMethodMapping#lookupHandlerMethod 方法就是根据 Request 的请求路径找到 HandlerMethaod,吐如果根据一一系列的匹配规则还是匹配不到,就给出匹配不到的提示,这里我们重点关注 this.handleMatch(bestMatch.mapping, lookupPath, request) 这行代码,这里面有对请求路径后拼接参数的处理。

//org.springframework.web.servlet.handler.AbstractHandlerMethodMapping#lookupHandlerMethod
//根据 urlpath 和 request 寻找具体的 HandlerMethod
@Nullable
protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {List<AbstractHandlerMethodMapping<T>.Match> matches = new ArrayList();//根据 url 从 mappingRegistry 中获取 RequestMappingInfoList<T> directPathMatches = this.mappingRegistry.getMappingsByDirectPath(lookupPath);if (directPathMatches != null) {//不为空  添加到 matches 中this.addMatchingMappings(directPathMatches, matches, request);}//matches 为空判断if (matches.isEmpty()) {//根据 urlpath 匹配到的结果为空 就将所有的映射关系加入 matchesthis.addMatchingMappings(this.mappingRegistry.getRegistrations().keySet(), matches, request);}//再次为空判断if (matches.isEmpty()) {//为空 处理return this.handleNoMatch(this.mappingRegistry.getRegistrations().keySet(), lookupPath, request);} else {//获取第一个AbstractHandlerMethodMapping<T>.Match bestMatch = (AbstractHandlerMethodMapping.Match)matches.get(0);if (matches.size() > 1) {//排序Comparator<AbstractHandlerMethodMapping<T>.Match> comparator = new AbstractHandlerMethodMapping.MatchComparator(this.getMappingComparator(request));matches.sort(comparator);//获取排序后的第一个bestMatch = (AbstractHandlerMethodMapping.Match)matches.get(0);if (this.logger.isTraceEnabled()) {this.logger.trace(matches.size() + " matching mappings: " + matches);}//是否是跨域请求if (CorsUtils.isPreFlightRequest(request)) {Iterator var7 = matches.iterator();while(var7.hasNext()) {AbstractHandlerMethodMapping<T>.Match match = (AbstractHandlerMethodMapping.Match)var7.next();//有跨域配置if (match.hasCorsConfig()) {//返回跨域配置的 handlermethodreturn PREFLIGHT_AMBIGUOUS_MATCH;}}} else {//普通请求 获取匹配到的第一个AbstractHandlerMethodMapping<T>.Match secondBestMatch = (AbstractHandlerMethodMapping.Match)matches.get(1);if (comparator.compare(bestMatch, secondBestMatch) == 0) {//比较有多个匹配的则抛出异常Method m1 = bestMatch.getHandlerMethod().getMethod();Method m2 = secondBestMatch.getHandlerMethod().getMethod();String uri = request.getRequestURI();throw new IllegalStateException("Ambiguous handler methods mapped for '" + uri + "': {" + m1 + ", " + m2 + "}");}}}//将匹配到的HandlerMethod 设置到 request 中request.setAttribute(BEST_MATCHING_HANDLER_ATTRIBUTE, bestMatch.getHandlerMethod());//获取到的请求 mapping 进行处理 主要是针对pattern类型进行 请求路径 url 和请求参数的解析 存放到requestthis.handleMatch(bestMatch.mapping, lookupPath, request);return bestMatch.getHandlerMethod();}
}

RequestMappingInfoHandlerMapping#handleMatch方法源码分析

RequestMappingInfoHandlerMapping#handleMatch 方法主要是对获取到的 RequestMappingInfo 对象中的方法的请求路径和参数进行处理,并设置到 Request 属性值中,这里主要区分了请求路径后面拼接参数和不拼接参数两种情况的处理,重点关注this.extractMatchDetails 方法。

//org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping#handleMatch
protected void handleMatch(RequestMappingInfo info, String lookupPath, HttpServletRequest request) {//调用父类的方法存储 urlpath super.handleMatch(info, lookupPath, request);//获取活跃的 Pattern 条件RequestCondition<?> condition = info.getActivePatternsCondition();//是否是 PathPatternsRequestCondition 类型if (condition instanceof PathPatternsRequestCondition) {//是  /order/queryorderthis.extractMatchDetails((PathPatternsRequestCondition)condition, lookupPath, request);} else {//否 例如 /order/queryorder/{id}this.extractMatchDetails((PatternsRequestCondition)condition, lookupPath, request);}//方法的 @RequestMapping 修饰的方法是否包含 produces 属性 if (!info.getProducesCondition().getProducibleMediaTypes().isEmpty()) {Set<MediaType> mediaTypes = info.getProducesCondition().getProducibleMediaTypes();//context-type 中的 mediaTypes 存储到requestrequest.setAttribute(PRODUCIBLE_MEDIA_TYPES_ATTRIBUTE, mediaTypes);}}

RequestMappingInfoHandlerMapping#extractMatchDetails 方法源码分析

RequestMappingInfoHandlerMapping#extractMatchDetails 方法主要是针对 url 上是否拼接变量值进行了处理,并把解析出来的属性设置给 Request。

//org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping#extractMatchDetails
private void extractMatchDetails(PathPatternsRequestCondition condition, String lookupPath, HttpServletRequest request) {PathPattern bestPattern;Map uriVariables;//路径模式请求条件 是否为空if (condition.isEmptyPathMapping()) {//为空 获取第一个 patternbestPattern = condition.getFirstPattern();//url 变量赋值为空uriVariables = Collections.emptyMap();} else {//获取解析后的 PathContainerPathContainer path = ServletRequestPathUtils.getParsedRequestPath(request).pathWithinApplication();//获取第一个bestPattern = condition.getFirstPattern();//根据 PathContainer 进行匹配PathMatchInfo result = bestPattern.matchAndExtract(path);Assert.notNull(result, () -> {return "Expected bestPattern: " + bestPattern + " to match lookupPath " + path;});//获取 url 上的变量uriVariables = result.getUriVariables();//作为属性设置给 Requestrequest.setAttribute(MATRIX_VARIABLES_ATTRIBUTE, result.getMatrixVariables());}//设置到 requestrequest.setAttribute(BEST_MATCHING_PATTERN_ATTRIBUTE, bestPattern.getPatternString());request.setAttribute(URI_TEMPLATE_VARIABLES_ATTRIBUTE, uriVariables);
}//org.springframework.web.servlet.mvc.method.RequestMappingInfoHandlerMapping#extractMatchDetails
private void extractMatchDetails(PatternsRequestCondition condition, String lookupPath, HttpServletRequest request) {String bestPattern;Map uriVariables;if (condition.isEmptyPathMapping()) {//为空直接赋值为 urlpathbestPattern = lookupPath;//变量赋值为空uriVariables = Collections.emptyMap();} else {//不为空bestPattern = (String)condition.getPatterns().iterator().next();//获取变量值 {id} 对应的值uriVariables = this.getPathMatcher().extractUriTemplateVariables(bestPattern, lookupPath);if (!this.getUrlPathHelper().shouldRemoveSemicolonContent()) {//作为属性设置给 Requestrequest.setAttribute(MATRIX_VARIABLES_ATTRIBUTE, this.extractMatrixVariables(request, uriVariables));}//对路径进行编码uriVariables = this.getUrlPathHelper().decodePathVariables(request, uriVariables);}//设置到 requestrequest.setAttribute(BEST_MATCHING_PATTERN_ATTRIBUTE, bestPattern);request.setAttribute(URI_TEMPLATE_VARIABLES_ATTRIBUTE, uriVariables);
}

AbstractHandlerMapping#getHandlerExecutionChain 方法源码分析

AbstractHandlerMapping#getHandlerExecutionChain 方法的主要作用就是构造一个 HandlerExecutionChain,会把传入的 handler 和所有拦截器通过责任链模式构造成一个 HandlerExecutionChain,当调用这个 handler 时就会通过这个责任链执行拦截器内的处理方法。

//org.springframework.web.servlet.handler.AbstractHandlerMapping#getHandlerExecutionChain
protected HandlerExecutionChain getHandlerExecutionChain(Object handler, HttpServletRequest request) {//handler 是否是  HandlerExecutionChain 类型 是 强转 不是 就用 handler 创建一个 HandlerExecutionChainHandlerExecutionChain chain = handler instanceof HandlerExecutionChain ? (HandlerExecutionChain)handler : new HandlerExecutionChain(handler);//获取适配器拦截器 循环遍历Iterator var4 = this.adaptedInterceptors.iterator();while(var4.hasNext()) {HandlerInterceptor interceptor = (HandlerInterceptor)var4.next();//interceptor 是否是映射器拦截器 MappedInterceptor if (interceptor instanceof MappedInterceptor) {//是 强转MappedInterceptor mappedInterceptor = (MappedInterceptor)interceptor;//是否能匹配到当前 Requestif (mappedInterceptor.matches(request)) {//是 添加到 chain中chain.addInterceptor(mappedInterceptor.getInterceptor());}} else {//不是 可能是普通拦截器 加入到 chain 中chain.addInterceptor(interceptor);}}//返回 chainreturn chain;
}

本文简单分析了 Spring MVC 工作流程中获取 Handler 的实现,整个过程先通过 Request 请求的一些属性,从整个 HandlerMapping 中获取到具体的 Handler,然后和当前请求应该使用的拦截器一起,通过责任链模式构造出一个拦截器链,看似是从 HandlerMapping 中获取处理当前请求的 Handler,实则最后返回的是一个拦截器链,希望本篇的细节剖析可以帮助大家建立更深的映像。

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

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/pingmian/21619.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

0603《哎选》已经稳定运行2年

0603《哎选》已经稳定运行2年 0603《哎选》已经稳定运行2年 介绍 2022年6月3日经过一年的努力&#xff0c;优雅草蜻蜓G系统原生版诞生&#xff0c;本产品应用于《哎选》&#xff0c;经过2年的运营不断的更新迭代&#xff0c;目前产品已经有了一定的用户量&#xff0c;本产品…

MySQL详细安装教程

MySQL详细安装教程 目录&#xff1a; MySQL简介安装步骤 2.1 下载MySQL安装包 2.2 安装MySQL 2.3 配置MySQL 2.4 验证安装示例代码总结MySQL简介 MySQL是一个开源的关系型数据库管理系统&#xff0c;由瑞典MySQL AB公司开发。它使用结构化查询语言&#xff08;SQL&#xff09…

life diagnostics生物标志物检测试剂盒一SPARCL™ Kits

Life Diagnostics公司专注于研发和生产临床前研究以及动物医学诊断相关的ELISA试剂盒、纯化的生物标志物和抗体等产品。产品覆盖心血管疾病、急性期反应、免疫毒性和免疫学标志物等研究领域&#xff0c;主要应用于小鼠、大鼠、兔、猫、狗、猪、猴和鸡等物种。 新产品——SPARCL…

Spark 3.5.1 升级 Java 17 异常 cannot access class sun.nio.ch.DirectBuffer

异常说明 使用Spark 3.5.1 升级到Java17的时候会有一个异常&#xff0c;异常如下 SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder". SLF4J: Defaulting to no-operation (NOP) logger implementation SLF4J: See http://www.slf4j.org/codes.htm…

go slice切片的详细知识(包含底层扩容)——2

目录 例子 例3&#xff1a;使用append逐个添加元素和一次性添加多个元素的区别 例4&#xff1a;order[low:high:max] 例5&#xff1a;当容量大于1024的时候&#xff0c;每次扩容真的是1.25倍吗&#xff1f; 本文是对上一篇文章的补充&#xff1a; go slice切片的详细知识&…

离轴磁编案例分享 - 机器人关机模组

客户产品 六轴协作机器人产品 关机模组 关机模组内部结构 项目介绍 客户需求: 需要离轴&#xff0c;优点&#xff1a;可以中空走线&#xff0c;方便线缆从机器人中间穿过去&#xff0c;可以更好得保护好线缆&#xff0c;不需要把线漏在外面&#xff0c;影响使用和产品寿命。目…

最适合上班族和宝妈的兼职副业,一天500多,小众副业项目

近年来&#xff0c;地方特色小吃逐渐受到人们的热烈追捧&#xff0c;尤其是在直播的助力下&#xff0c;许多地方的特色小吃得以走进大众视野&#xff0c;吸引了大量流量和人气。因此&#xff0c;有很大一部分商家和创业者看准了这一商机&#xff0c;纷纷投身于地方特色小吃的制…

怎么把多种内容做成二维码?扫码展现多种内容的制作方法

现在很多的场景下都有不同类型的二维码&#xff0c;用来承载内容为用户提供内容展示&#xff0c;比如图片、视频、文字、文件、地图等等内容&#xff0c;都可以组合起来通过扫码的方式在手机上展示。那么如何制作组合内容的二维码相信有很多的小伙伴都非常的感兴趣。 其实二维…

众汇:外汇狙击指标如何使用?

对于投资者来说&#xff0c;我们各位交易的目的是什么?WeTrade众汇认为那就是盈利。所以来说有一个指标对各位投资者来说那是相当有帮助的。这是因为对于交易者而言&#xff0c;利用这些指标可以快速识别盈利的买卖时机。当我们选择一个指标之后&#xff0c;深入了解其适用范围…

【SpringBoot】打包成Docker镜像后日志输出中文乱码

解决方法 配置文件中对日志的配置添加如下选项 logging:charset:file: UTF-8console: UTF-8注:如果只需要解决控制台乱码,则不需要file这一项

「布道师系列文章」众安保险王凯解析 Kafka 网络通信

作者&#xff5c;众安保险基础平台 Java 开发专家王凯 引言 今天给大家带来的是 Kafka 网路通信主要流程的解析&#xff08;基于 Apache Kafka 3.7[2]&#xff09;。同时引申分析了业界当前较火的AutoMQ基于Kafka在网络通信层面的优化和提升。 01 如何构建一个基本的请求…

学习笔记(一)——Langchain基本操作与函数

学习笔记(一)——Langchain基本操作与函数 目录 学习笔记(一)——Langchain基本操作与函数基本初始化配置LangsmithLanguage Models 基础指令传递信息OutputParsers 输出解析器chain 链Prompt Templates 提示模板Message History 消息历史记录Managing Conversation History 管…

【机器学习】之 kmean算法原理及实现

基本概念 K-Means 聚类算法的目标是将数据集分成 ( K ) 个簇&#xff0c;使得每个簇内的数据点尽可能相似&#xff0c;而簇与簇之间尽可能不同。这种相似度是通过计算数据点与簇中心的距离来衡量的。 算法步骤 选择簇的数量 ( K )&#xff1a;随机选择 ( K ) 个数据点作为初…

XL7005A SOP-8 0.4A1.25-20V 150KHz降压直流转换器芯片

XL7005A作为一款高性能的降压型电源管理芯片&#xff0c;在智能家居中有着广泛的应用。以下是一些具体的案例&#xff1a; 1. 智能灯具&#xff1a;XL7005A可用于控制LED灯的电源&#xff0c;提供稳定高效的电源支持&#xff0c;确保灯具亮度稳定且无频闪&#xff0c;提高用户体…

springboot从2.7.2 升级到 3.3.0

文章目录 概要准备报错调整小结后记 概要 时代在进步&#xff0c;springboot已经来到了3.3.0 , 于是我们也打算升级下sbvadmin到3.3&#xff0c; jdk使用21的版本&#xff0c;下面是升级过程中碰到的一些问题&#xff0c;问题不大。 2.7.2 -> 3.3.0 准备 下载jdk21&#…

接口框架项目实战-pytest(六)csv数据驱动

csv 数据驱动 为了解决数据量大 导致yaml文件重复太多 yaml_util.py import osimport jsonpath import yamlfrom pytestdemo.common.base_util import get_path from pytestdemo.common.csv_util import analysis_parametersdef read_config_file(one_node,two_node):with ope…

windows的软件修改图标

要修改一个可执行文件&#xff08;.exe&#xff09;的图标&#xff0c;你可以使用 Resource Hacker 这样的工具。Resource Hacker 是一个免费的资源编辑器&#xff0c;可以用于修改和编辑 Windows 可执行文件中的资源。 以下是一个简单的步骤来修改一个 .exe 文件的图标&#x…

shell脚本 字符串拼接变量赋值失效

问题现象&#xff1a; 代码如下&#xff1a; 执行结果&#xff1a; 可以看到data_dir属性是有值的&#xff0c;但是做字符串拼接变量赋值失效了很奇怪 怀疑赋值哪里写错了 问题分析&#xff1a; 1. 还是觉得赋值没有问题&#xff0c;手动显式赋值再执行下 执行结果&#…

职场如同“染缸”,老板只是给你个平台,染的好坏,全凭运气!

无论哪个单位&#xff0c;在职场大染缸里总有那么一拨同事是你喜欢的&#xff0c;也有那么一拨同事是不痛不痒的&#xff0c;还有那么一拨同事却是你怎么看都觉得不顺眼的。“不顺眼”的定义很宽泛&#xff0c;可能是他曾经的一些言论触及了你的道德底线&#xff0c;可能是他的…

100vh问题及解决方案

100vh 问题通常出现在移动端浏览器中&#xff0c;尤其是当你使用 100vh 设定元素高度时。这是因为移动端浏览器在显示视口高度时会包括地址栏和工具栏&#xff0c;这些栏在滚动时会隐藏&#xff0c;从而导致视口高度发生变化&#xff0c;影响布局。 具体问题 高度不稳定&…