RequestMappingHandlerMapping 详解
我们先理简单梳理一个关系
关系梳理
- spring ioc 是spring的核心,用来管理spring bean的生命周期
 - MVC 是一种使用 MVC(Model View Controller 模型-视图-控制器)设计创建 Web 应用程序的模式
 - spring mvc 是spring的一个独立的模块,就像AOP一样
 
在spring mvc中把web框架和spring ioc融合在一起,是通过ContextLoaderListener监听servlet上下文的创建后来加载父容器完成的,然后通过配置一个servlet对象DispatcherServlet,在初始化DispatcherServlet时来加载具体子容器,详细的可以参考spring ioc & web宿主 这篇文章
关于我们今天要讲的RequestMappingHandlerMapping也是在DispatcherServlet的初始化过程中自动加载的,默认会自动加载所有实现HandlerMapping接口的bean,且我们可以通过serOrder来设置优先级,系统默认会加载RequestMappingHandlerMapping、BeanNameUrlHandlerMapping、SimpleUrlHandlerMapping 并且按照顺序使用
 1private void initHandlerMappings(ApplicationContext context) {
 2        this.handlerMappings = null;
 3        if (this.detectAllHandlerMappings) {
 4            // Find all HandlerMappings in the ApplicationContext, including ancestor contexts.
 5            Map<String, HandlerMapping> matchingBeans =
 6                    BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
 7            if (!matchingBeans.isEmpty()) {
 8                this.handlerMappings = new ArrayList<>(matchingBeans.values());
 9                // We keep HandlerMappings in sorted order.
10                AnnotationAwareOrderComparator.sort(this.handlerMappings);
11            }
12        }
13}
RequestMappingHandlerMapping 加载过程
- RequestMappingHandlerMapping 实现了接口InitializingBean,在bean加载完成后会自动调用afterPropertiesSet方法,在此方法中调用了initHandlerMethods()来实现初始化
 - 遍历所有bean,如果bean实现带有注解@Controller或者@RequestMapping 则进一步调用detectHandlerMethods处理,处理逻辑大致就是根据@RequestMapping配置的信息,构建RequestMappingInfo,然后注册到MappingRegistry中
 
 1protected void initHandlerMethods() {
 2        String[] beanNames = (this.detectHandlerMethodsInAncestorContexts ?
 3                BeanFactoryUtils.beanNamesForTypeIncludingAncestors(obtainApplicationContext(), Object.class) :
 4                obtainApplicationContext().getBeanNamesForType(Object.class));
 5        for (String beanName : beanNames) {
 6            if (!beanName.startsWith(SCOPED_TARGET_NAME_PREFIX)) {
 7                Class<?> beanType = null;
 8                beanType = obtainApplicationContext().getType(beanName);
 9                if (beanType != null && isHandler(beanType)) {
10                    detectHandlerMethods(beanName);
11                }
12            }
13        }
14        handlerMethodsInitialized(getHandlerMethods());
15    }
 1protected void detectHandlerMethods(final Object handler) {
 2        Class<?> handlerType = (handler instanceof String ?
 3                obtainApplicationContext().getType((String) handler) : handler.getClass());
 4        if (handlerType != null) {
 5            final Class<?> userType = ClassUtils.getUserClass(handlerType);
 6            Map<Method, T> methods = MethodIntrospector.selectMethods(userType,
 7                    (MethodIntrospector.MetadataLookup<T>) method -> {
 8                        try {
 9                            return getMappingForMethod(method, userType);
10                        }
11                        catch (Throwable ex) {
12                            throw new IllegalStateException("Invalid mapping on handler class [" +
13                                    userType.getName() + "]: " + method, ex);
14                        }
15                    });
16            methods.forEach((method, mapping) -> {
17                Method invocableMethod = AopUtils.selectInvocableMethod(method, userType);
18                registerHandlerMethod(handler, invocableMethod, mapping);
19            });
20        }
21    }
RequestMappingHandlerMapping 解析过程
- 在DispatcherServlet中,根据请求对象调用getHander方法获取HandlerExecutionChain对象
 - 在getHander方法中也是遍历上面默认加载的三个HandlerMapping,当然第一个就是RequestMappingHandlerMapping对象,调用其getHandler方法,根据请求path,找到一个最为匹配的HandlerMethod来处理请求
 
 1protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
 2        if (this.handlerMappings != null) {
 3            for (HandlerMapping hm : this.handlerMappings) {
 4                if (logger.isTraceEnabled()) {
 5                    logger.trace(
 6                            "Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'");
 7                }
 8                HandlerExecutionChain handler = hm.getHandler(request);
 9                if (handler != null) {
10                    return handler;
11                }
12            }
13        }
14        return null;
15    }
- 根据请求路径获取HandlerInterceptor,然后和上面获得的HandlerMethod一起构成HandlerExecutionChain返回给DispatcherServlet
 
DispatcherServlet得到HandlerExecutionChain也就获得了处理此次请求所需的Handler【即我们熟悉的Controller和对应的Action】,后续将会选择合适HandlerAdapter来执行对应的Handler,获取返回值,再根据返回值类型,进一步觉决定用什么方式展示给用户,下一遍将开启HandlerAdapter的讲解…….
微信公众号:宋坤明
更多精彩请参考 完整版系列 也可以直接关注我
图注:宋坤明公众号
RequestMappingHandlerMapping 详解的更多相关文章
- RequestMappingHandlerMapping详解
		
我们先理简单梳理一个关系 关系梳理 spring ioc 是spring的核心,用来管理spring bean的生命周期 MVC 是一种使用 MVC(Model View Controller 模型- ...
 - spring-mvc注解(mvc:annotation-driver,JSON,配置详解)
		
一.DefaultAnnotationHandlerMapping 和 AnnotationMethodHandlerAdapter 的使用已经过时! spring 3.1 开始我们应该用 Reque ...
 - Spring MVC 学习总结(二)——控制器定义与@RequestMapping详解
		
一.控制器定义 控制器提供访问应用程序的行为,通常通过服务接口定义或注解定义两种方法实现. 控制器解析用户的请求并将其转换为一个模型.在Spring MVC中一个控制器可以包含多个Action(动作. ...
 - SpringMVC异常处理机制详解[附带源码分析]
		
目录 前言 重要接口和类介绍 HandlerExceptionResolver接口 AbstractHandlerExceptionResolver抽象类 AbstractHandlerMethodE ...
 - HandlerMapping 详解
		
HandlerMapping 详解 1. 导言 万丈高楼平地起,SpringMVC的辉煌离不开每个组件的相互协作,上一章详细阐述了SpringMVC整个体系结构及实现原理,知道HandlerMappi ...
 - 详解SpringMVC请求的时候是如何找到正确的Controller
		
详解SpringMVC请求的时候是如何找到正确的Controller[附带源码分析] 目录 前言 源码分析 重要接口介绍 SpringMVC初始化的时候做了什么 HandlerExecutionCha ...
 - SpringMVC 框架系列之组件概述与配置详解
		
在上一篇文章 SpringMVC 框架系列之初识与入门实例 的实例中,我们已经知道,SpringMVC 框架是一个 web 层的框架,本篇文章就详细解释一下 SpringMVC 框架具体文件的配置以及 ...
 - SpringMVC【开发Controller】详解
		
前言 本文主要是讲解在Controller中的开发,主要的知识点有如下: 编码过滤器 使用注解开发 注解@RequestMapping详解 业务方法接收参数 字符串转日期 重定向和转发 返回JSON ...
 - 转载 Spring、Spring MVC、MyBatis整合文件配置详解
		
Spring.Spring MVC.MyBatis整合文件配置详解 使用SSM框架做了几个小项目了,感觉还不错是时候总结一下了.先总结一下SSM整合的文件配置.其实具体的用法最好还是看官方文档. ...
 
随机推荐
- CentOS 7.2搭建xl2tp服务器
			
## 1.下载xl2tpd.tar.gz源码包 ```wget http://pkgs.fedoraproject.org/repo/pkgs/xl2tpd/xl2tpd-1.3.8.tar.gz/d ...
 - Activemq首次运行报错 “找不到或无法加载主类”
			
首次运行Program Files\apache-activemq-5.10.0\bin目录下的activemq.bat文件,报错信息如下: 找不到或无法加载主类 Files\apache-activ ...
 - 20155217 2016-2017-2 《Java程序设计》第3周学习总结
			
20155217 2016-2017-2 <Java程序设计>第3周学习总结 教材学习内容总结 第四章 要产生对象必须先定义类,类定义时使用class关键词,建立实例要使用new关键词. ...
 - 20155229 2016-2017-2 《Java程序设计》第二周学习总结
			
20155229 2016-2017-2 <Java程序设计>第二周学习总结 教材学习内容总结 布尔:boolean类型可表示true和false %符号被用来作为控制符号前置,所以规定用 ...
 - 20155233 2006-2007-2 《Java程序设计》第4周学习总结
			
20155233 2006-2007-2 <Java程序设计>第4周学习总结 教材学习内容总结 第六章: 继承:子类继承父类,避免重复的行为定义. extends关键词:继承并扩充行为. ...
 - 20155308 2016-2017-2《Java程序设计》课堂实践项目
			
20155308 2016-2017-2<Java程序设计>课堂实践项目 在java.lang包中有String.split()方法,返回是一个数组 我在应用中用到一些,给大家总结一下,仅 ...
 - 20155323 第四次实验 Android程序设计实验报告
			
20155323 第四次实验 Android程序设计实验报告 实验内容 1.基于Android Studio开发简单的Android应用并部署测试; 2.了解Android.组件.布局管理器的使用: ...
 - primary key和unique的区别
			
定义了 UNIQUE 约束的字段中不能包含重复值,可以为一个或多个字段定义 UNIQUE 约束.因此,UNIQUE 即可以在字段级也可以在表级定义, 在 UNIQUED 约束的字段上可以包含空值.OR ...
 - Spring之HandlerInterceptor拦截器
			
思维导图下载:https://pan.baidu.com/s/19z73Bs8MsHFAupga3Cr3Gg
 - kali安装后相关软件的配置
			
更新软件apt-get updateapt-get upgrade安装输入法apt-get install ibus-pinyin apt-get install netspeed安装GNOMEapt ...