2023-09-14  阅读(2)
原文作者:王伟王胖胖 原文地址: https://blog.csdn.net/wangwei19871103/article/details/105523125

简单流程图

202309142300528001.png

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);
    	}

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

202309142300539983.png

202309142300546434.png
然后是类上的:

202309142300553695.png

combine合并属性

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

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

202309142300574107.png
最后放入methodMap返回:

202309142300581668.png

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注解相关处理。

202309142300591099.png

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.");
    			}
    		}

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


Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。

它的内容包括:

  • 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
  • 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
  • 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
  • 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
  • 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
  • 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
  • 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
  • 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw

目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:

想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询

同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。

阅读全文