Spring MVC源码解析:HandlerMapping处理器映射器详解
创作时间:
作者:
@小白创作中心
Spring MVC源码解析:HandlerMapping处理器映射器详解
引用
CSDN
1.
https://blog.csdn.net/Lxn2zh/article/details/128545470
本文将深入分析Spring MVC框架中的HandlerMapping处理器映射器的源码实现。从概念到具体实现,逐步展开,帮助读者理解Spring MVC的请求处理流程。
系列文章目录
- springmvc源码之Web上下文初始化
- springmvc源码之DispatcherServlet前端控制器
- springmvc源码之HandlerMapping处理器映射器
HandlerMapping处理器映射器
HandlerMapping的主要作用是根据请求(request)找到相应的处理器(Handler)和拦截器(Interceptors)。其核心方法如下:
HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception;
实现类
HandlerMapping帮助DispatcherServlet进行web请求的url到具体处理类的匹配,用来根据请求的url查找Handler,内部维护的Map<String, Object>映射,springmvc提供了不同的映射器实现不同的映射方式,例如:配置文件方式,实现接口方式,注解方式等
private final Map<String, Object> handlerMap = new LinkedHashMap<String, Object>();
Spring自带了多个处理器映射实现:
- BeanNameUrlHandlerMapping:根据控制器Bean的名字将控制器映射到URL
- ControllerBeanNameHandlerMapping:与BeanNameUrlHandlerMapping类似
- ControllerClassNameHandlerMapping:通过使用控制器的类名作为URL基础将控制器映射到URL
- DefaultAnnotationHandlerMapping:将请求映射给使用@RequestMapping注解的控制器和控制器方法
- SimpleUrlHandlerMapping:使用定义在Spring应用上下文的集合将控制器映射到URL
- RequestMappingHandlerMapping:SpringMVC3.1新增的,在springMVC3.1之前,DefaultAnnotationHandlerMapping会在类级别上选中一个控制器,然后通过AnnotationMethodHandlerAdapter定位到具体要调用的方法;而在SpringMVC3.1之后,这些操作全都放生在RequestMappingHandlerMapping中,从类级别和方法级别的@RequestMapping注解中获取到路径映射信息,使得在HandlerInterceptor中获取到的处理器肯定是一个HandlerMethod类型
配置
<!-- 开启注解 -->
<mvc:annotation-driven/>
<bean id="defaultAnnotationHandlerMapping"
class="org.springframework.web.servlet.mvc.annotation.AnnotationMethodHandlerAdapter"/>
<mvc:annotation-driven/>配置的作用:
- 自动注册RequestMappingHandlerMapping、RequestMappingHandlerAdater、ExceptionHandlerExceptionResolver三个bean
- 支持使用ConversionService实例对表单参数进行类型转换
- 支持使用@NumberFormatannotation、@DataTimeFormat注解完成数据类型的格式化
- 支持使用@Vaild注解对JavaBean实例进行JSR 303验证
- 支持使用@RequestBody和@ResponseBody注解
RequestMappingHandlerMapping源码
由于一般都使用<mvc:annotation-driven/>进行配置,所以就以RequestMappingHandlerMapping为例进行讲解
创建
首先进行创建
其实现了ApplicationContextAware,所以需要执行setApplicationContext
// 调用链路 org.springframework.context.support.ApplicationObjectSupport#setApplicationContext ——>org.springframework.context.support.ApplicationObjectSupport#initApplicationContext(org.springframework.context.ApplicationContext) -->org.springframework.web.servlet.handler.AbstractHandlerMapping#initApplicationContext
// org.springframework.web.servlet.handler.AbstractHandlerMapping#initApplicationContext
protected void initApplicationContext() throws BeansException {
// 扩展interceptors的方法,空实现
extendInterceptors(this.interceptors);
// 将容器中的所有MappedInterceptor类型的bean添加到mappedInterceptors中
detectMappedInterceptors(this.adaptedInterceptors);
// 初始化Interceptor,将interceptors中的对象添加到adaptedInterceptors中
initInterceptors();
}
其实现了InitializingBean,需要执行afterPropertiesSet
// org.springframework.web.servlet.handler.AbstractHandlerMethodMapping#afterPropertiesSet
public void afterPropertiesSet() {
initHandlerMethods();
}
// org.springframework.web.servlet.handler.AbstractHandlerMethodMapping#initHandlerMethods
protected void initHandlerMethods() {
// 拿到容器中的bean,筛选出Handler
String[] beanNames = (this.detectHandlerMethodsInAncestorContexts ?
BeanFactoryUtils.beanNamesForTypeIncludingAncestors(getApplicationContext(), Object.class) :
getApplicationContext().getBeanNamesForType(Object.class));
for (String beanName : beanNames) {
if (!beanName.startsWith(SCOPED_TARGET_NAME_PREFIX)) {
Class<?> beanType = null;
try {
beanType = getApplicationContext().getType(beanName);
}
catch (Throwable ex) {
}
// (AnnotatedElementUtils.hasAnnotation(beanType, Controller.class) ||AnnotatedElementUtils.hasAnnotation(beanType, RequestMapping.class)); 判断
if (beanType != null && isHandler(beanType)) {
// 将访问地址、Method进行映射
detectHandlerMethods(beanName);
}
}
}
handlerMethodsInitialized(getHandlerMethods());
}
访问
在进行访问的时候会通过org.springframework.web.servlet.DispatcherServlet#getHandler方法来遍历handlerMappings
HandlerExecutionChain handler = hm.getHandler(request);
来调用HandlerMapping的getHandler方法
// org.springframework.web.servlet.handler.AbstractHandlerMapping#getHandler
public final HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
// 找到Handler,根据地址找到对应的方法
Object handler = getHandlerInternal(request);
if (handler == null) {
handler = getDefaultHandler();
}
if (handler == null) {
return null;
}
// Bean name or resolved handler?
if (handler instanceof String) {
String handlerName = (String) handler;
handler = getApplicationContext().getBean(handlerName);
}
// 为handler生成执行链,即添加interceptor
HandlerExecutionChain executionChain = getHandlerExecutionChain(handler, request);
if (CorsUtils.isCorsRequest(request)) {
CorsConfiguration globalConfig = this.globalCorsConfigSource.getCorsConfiguration(request);
CorsConfiguration handlerConfig = getCorsConfiguration(handler, request);
CorsConfiguration config = (globalConfig != null ? globalConfig.combine(handlerConfig) : handlerConfig);
executionChain = getCorsHandlerExecutionChain(request, executionChain, config);
}
return executionChain;
}
getHandlerInternal
// org.springframework.web.servlet.handler.AbstractHandlerMethodMapping#getHandlerInternal
protected HandlerMethod getHandlerInternal(HttpServletRequest request) throws Exception {
String lookupPath = getUrlPathHelper().getLookupPathForRequest(request);
if (logger.isDebugEnabled()) {
logger.debug("Looking up handler method for path " + lookupPath);
}
this.mappingRegistry.acquireReadLock();
try {
HandlerMethod handlerMethod = lookupHandlerMethod(lookupPath, request);
if (logger.isDebugEnabled()) {
if (handlerMethod != null) {
logger.debug("Returning handler method [" + handlerMethod + "]");
}
else {
logger.debug("Did not find handler method for [" + lookupPath + "]");
}
}
return (handlerMethod != null ? handlerMethod.createWithResolvedBean() : null);
}
finally {
this.mappingRegistry.releaseReadLock();
}
}
getHandlerExecutionChain
protected HandlerExecutionChain getHandlerExecutionChain(Object handler, HttpServletRequest request) {
HandlerExecutionChain chain = (handler instanceof HandlerExecutionChain ?
(HandlerExecutionChain) handler : new HandlerExecutionChain(handler));
String lookupPath = this.urlPathHelper.getLookupPathForRequest(request);
for (HandlerInterceptor interceptor : this.adaptedInterceptors) {
if (interceptor instanceof MappedInterceptor) {
MappedInterceptor mappedInterceptor = (MappedInterceptor) interceptor;
if (mappedInterceptor.matches(lookupPath, this.pathMatcher)) {
chain.addInterceptor(mappedInterceptor.getInterceptor());
}
}
else {
chain.addInterceptor(interceptor);
}
}
return chain;
}
参考文献
- HandlerMapping
热门推荐
中国历史上规模最大的战役——长平之战
四大百人斩武将
全球学子嘉游赣|南昌上榜同程长江国际黄金旅游带秋冬畅游人气榜单
广州塔必打卡:小蛮腰的浪漫之旅
如何在聊天中有效地表达自己的情感?
中国传统文化:照亮现代生活的智慧之光
鬼谷子:邯郸历史名人的智慧传承
邯郸城内:赵武灵王的胡服骑射传奇
揭秘眼镜蛇的古代称呼:从“过山风”到“神圣毒蛇”
揭秘印度神话:眼镜蛇的神秘别称“那伽”
眼镜蛇古称"过山风"?揭秘这条"多面蛇"的古今称谓
史上搞笑死法大揭秘:从汉斯被胡子绊倒说起
古代帝王的荒唐死法:被绿与醉酒的悲剧
黑曼巴蛇:非洲最具威胁性的毒蛇
威胁人类的七种"小动物",最后一种每年致死数十万人
大理旅游住宿价格分析及性价比推荐
上海激发首个非遗春节新活力
睡前一杯热牛奶,助你一夜好眠
失眠救星!酸枣仁汤&百合莲子银耳汤
英国睡眠专家盖伊·梅多斯的九大睡眠改善技巧
“还阳卧”真的能助眠吗?真相揭秘
广西美食如何香飘更远?
大理丽江:云南文化探秘之旅
五一必打卡:云南石林、洱海、玉龙雪山
五一打卡云南必游景点:玉龙雪山、洱海、石林
四月底云南游,找导游婷婷不踩坑!
镇远古镇vs铁溪景区:谁才是你的菜?
镇远古镇:穿越千年的历史与酸汤鱼的诱惑
探访东方威尼斯:镇远古镇的文化遗产守护者
跟着赛事去旅行 在冰雪中享受运动乐趣——冬季到四川来旅行(中)