SpringMvc之映射器HandlerMapping

简介

在springmvc的处理流程中,第一步就是查询请求对应的映射器,然后组装成处理器链处理请求,本文意在梳理该过程

重要实现

HandlerMapping是一个接口,该接口用于通过HttpServletRequest寻找对应的处理器,接口介绍如下:

该接口只有一个抽象方法:

该接口的几个重要子类和关系图如下:

在HandlerMapping的直接子类:AbstractHandlerMapping中有定义核心方法:

后续的各种HandlerMapping均是扩展该接口实现自身逻辑,以下以最常用的关于注解:@RestController,@Controller,@RequestMapping使用的处理器映射关系为例做源码分析

RequestMappingHandlerMapping

类的UML图如下:

由类的UML图可知,RequestMappingHandlerMapping集成了AbstractHandlerMethodMapping,

而AbstractHandlerMethodMapping类定义如下:

所以:RequestMappingHandlerMapping类也实现了InitializingBean,在类的初始化创建完成之后,会做相应的初始化操作,具体的代码如下:

方法initHandlerMethods()即是初始化handler方法,调用processCandidateBean完成ioc的bean对象处理:

其中IsHandler方法用于检测该bean对象是否符合条件,实现交RequestMappingHandlerMapping实现

判断标准就是该类是否有标注@Controller(@RestController注解有标注@Controller)注解或者@RequestMapping注解

符合条件的类继续往下对方法做探测:

对该类的每个方法均做检测,看看是否符合条件,而符合条件的方法检测标准即是是否有标注@RequestMapping注解(@GetMapping等注解均被@RequestMapping注解标注),这里具体的实现逻辑在org.springframework.web.servlet.mvc.method.annotation.RequestMappingHandlerMapping#getMappingForMethod

当所有符合条件的方法检测完成之后,需要对对应的方法和组装的RequestMappingInfo(可以是其他对象)完成注册操作

MappingRegistry

该类是AbstractHandlerMethodMapping的内部类,该类中定义多个缓存对象,用于存放各种映射关系,如下图所示:

其中有两个重要的缓存对象:

复制代码
mappingLookup:用于存放RequestMappingInfo(可以是其他对象)和HandlerMethod的映射关系
复制代码
urlLookup:用于存放请求路径和RequestMappInfo(可以是其他对象)的映射关系

至此:RequestMappingHandlerMapping的启动流程分析完毕,总结如下:

RequestMappingHandlerMapping在初始化完成之后,调用父类AbstractHandlerMethodMapping的initHandlerMethods方法,该方法会检测ioc中所有符合条件的(@Controller@RequestMapping的类),然后依次检测该类中标注了@RequestMapping的方法,然后组装成RequestMappingInfo对象,将方法和RequestMappingInfo的映射关系注册到MappingRegistry的两个缓存对象:

复制代码
mappingLookup、urlLookup中,完成注册操作

而在实际请求中则是以上过程的逆过程,相关源码感兴趣的同学可以自行调试分析

自定义端点源码分析

在springboot actuator中有很多的监控信息可以通过暴露"端点"的方法让外部获取应用的信息,而端点的暴露即和以上过程大体相当

在图3所示的springboot actuator的自定义端点是通过WebMvcEndpointHandlerMapping暴露的

从图3的类继承图可知:WebMvcEndpointHandlerMapping继承子AbstractHandlerMethodMapping

在类初始化完成之后,一样会调用afterPropertiesSet,AbstractWebMvcEndpointHandlerMapping重写了initHandlerMethods方法,具体的逻辑如下:

在这里是通过构建WebOperation暴露端点,同样的,将相关信息(RequestMappingInfo)和HandlerMethod的映射关系注册到AbstractHanderMethodMapping的内部类MappingRegistry中完成初始化过程

以下简单的梳理一下Actuator暴露端点的过程

1 spring-boot-starter-actuator依赖引入了依赖spring-boot-actuator-autoconfigure,通过springboot的自动装配原理,会读取spring.factories的类

2 WebMvcEndpointManagementContextConfiguration配置类中通过条件配置引入了WebMvcEndpointHandlerMapping对象

3 WebMvcEndpointHandlerMapping执行初始化流程,完成RequestMappingInfo和HandlerMethod的映射注册

以下是引入springboot-actuator的能暴露的Endpoint类实现

具体的信息获取逻辑感兴趣的可以自行调试

相关推荐
ShineWinsu2 小时前
对于C++:继承的解析—上
开发语言·数据结构·c++·算法·面试·笔试·继承
小付同学呀3 小时前
C语言学习(五)——输入/输出
c语言·开发语言·学习
码农阿豪3 小时前
Nacos 日志与 Raft 数据清理指南:如何安全释放磁盘空间
java·安全·nacos
梦幻精灵_cq3 小时前
学C之路:不可或缺的main()主函数框架(Learn-C 1st)
c语言·开发语言
直有两条腿3 小时前
【大模型】Langchain4j
java·langchain
love530love3 小时前
Scoop 完整迁移指南:从 C 盘到 D 盘的无缝切换
java·服务器·前端·人工智能·windows·scoop
消失的旧时光-19433 小时前
C++ 多线程与并发系统取向(二)—— 资源保护:std::mutex 与 RAII(类比 Java synchronized)
java·开发语言·c++·并发
莫寒清3 小时前
ThreadLocal
java·面试
福大大架构师每日一题4 小时前
go-zero v1.10.0发布!全面支持Go 1.23、MCP SDK迁移、性能与稳定性双提升
开发语言·后端·golang
学习是生活的调味剂4 小时前
spring bean循环依赖问题分析
java·后端·spring