Spring Boot源码之旅二十四SpringMVC源码之RequestMappingHandlerMapping的初始化二

作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO

联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬

学习必须往深处挖,挖的越深,基础越扎实!

阶段1、深入多线程

阶段2、深入多线程设计模式

阶段3、深入juc源码解析


阶段4、深入jdk其余源码解析


阶段5、深入jvm源码解析

 

码哥源码部分

码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】

码哥讲源码【炸雷啦!炸雷啦!黄光头他终于跑路啦!】

码哥讲源码-【jvm课程前置知识及c/c++调试环境搭建】

 

​​​​​​码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】

码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】

码哥讲源码【你水不是你的错,但是你胡说八道就是你不对了!】

码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】

终结B站没人能讲清楚红黑树的历史,不服等你来踢馆!

打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】

简单流程图

RequestMappingHandlerMapping的createRequestMappingInfo创建映射

首先获取RequestMapping注解,如果存在就创建一个RequestMappingInfo否则就返回null 。

    	@Nullable
    	private RequestMappingInfo createRequestMappingInfo(AnnotatedElement element) {
    		RequestMapping requestMapping = AnnotatedElementUtils.findMergedAnnotation(element, RequestMapping.class);
    		RequestCondition<?> condition = (element instanceof Class ?
    				getCustomTypeCondition((Class<?>) element) : getCustomMethodCondition((Method) element));
    		return (requestMapping != null ? createRequestMappingInfo(requestMapping, condition) : null);
    	}


其实就是封装方法的RequestMappingInfo注解的属性啦:


然后是类上的:

combine合并属性

具体怎么合并的细节就不看了。


合并后的结果,我们常用的就是uri拼起来了:


最后放入methodMap返回:

registerHandlerMethod

注册方法,最终是注册到MappingRegistry中的。

    	@Override
    	protected void registerHandlerMethod(Object handler, Method method, RequestMappingInfo mapping) {
    		super.registerHandlerMethod(handler, method, mapping);
    		updateConsumesCondition(mapping, method);
    	}
    	protected void registerHandlerMethod(Object handler, Method method, T mapping) {
    		this.mappingRegistry.register(mapping, handler, method);
    	}

MappingRegistry的register

注册各种类型的映射。

    		private final Map<T, MappingRegistration<T>> registry = new HashMap<>();
    
    		private final Map<T, HandlerMethod> mappingLookup = new LinkedHashMap<>();
    
    		private final MultiValueMap<String, T> urlLookup = new LinkedMultiValueMap<>();
    
    		private final Map<String, List<HandlerMethod>> nameLookup = new ConcurrentHashMap<>();
    
    		public void register(T mapping, Object handler, Method method) {
    			...
    			this.readWriteLock.writeLock().lock();
    			try {
    				HandlerMethod handlerMethod = createHandlerMethod(handler, method);//创建HandlerMethod
    				validateMethodMapping(handlerMethod, mapping);//验证唯一性
    				this.mappingLookup.put(mapping, handlerMethod);//方法映射,放入mappingLookup
    
    				List<String> directUrls = getDirectUrls(mapping);
    				for (String url : directUrls) {
    					this.urlLookup.add(url, mapping);//url映射,放入mappingLookup,可以多对一
    				}
    
    				String name = null;
    				if (getNamingStrategy() != null) {
    					name = getNamingStrategy().getName(handlerMethod, mapping);
    					addMappingName(name, handlerMethod);//名字映射,name为类名大写字母+#+方法名,比如UserController的getUser方法就是UC#getUser
    				}
    				...
    				this.registry.put(mapping, new MappingRegistration<>(mapping, handlerMethod, directUrls, name));//封装成MappingRegistration放入registry
    			}
    			finally {
    				this.readWriteLock.writeLock().unlock();
    			}
    		}
AbstractHandlerMethodMapping的createHandlerMethod

根据是处理器是String还是对象类型进行不同的HandlerMethod封装,一般初始化的时候还没有实例,只是名字。

    	protected HandlerMethod createHandlerMethod(Object handler, Method method) {
    		if (handler instanceof String) {//如果是bean名字
    			return new HandlerMethod((String) handler,
    					obtainApplicationContext().getAutowireCapableBeanFactory(), method);
    		}
    		return new HandlerMethod(handler, method);
    	}

处理器名字创建:
内部会获取名字对应的类型,进行方法参数信息的封装和ResponseStatus注解相关处理。

MappingRegistry的validateMethodMapping检查唯一性

如果uri映射HandlerMethod已经存在,且不是同一个HandlerMethod要报异常。

    private void validateMethodMapping(HandlerMethod handlerMethod, T mapping) {
    			// Assert that the supplied mapping is unique.
    			HandlerMethod existingHandlerMethod = this.mappingLookup.get(mapping);
    			if (existingHandlerMethod != null && !existingHandlerMethod.equals(handlerMethod)) {
    				throw new IllegalStateException(
    						"Ambiguous mapping. Cannot map '" + handlerMethod.getBean() + "' method \n" +
    						handlerMethod + "\nto " + mapping + ": There is already '" +
    						existingHandlerMethod.getBean() + "' bean method\n" + existingHandlerMethod + " mapped.");
    			}
    		}

好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。

  • 28
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值