springMVC是如何做url映射到controller的?

背景

最近要做一个URL到权限的控制,需要根据实际的URL匹配到配置文件,一般的url好处理,但是对于path parameter 的URL就需要做匹配了,所以研读了一下springmvc的源码,看看是如何做的

  • 入口 FrameworkServlet

    • .doPost(HttpServletRequest request, HttpServletResponse response)
  • DispatcherServlet

    • .doService
    • DispatcherServlet.doDispatch
    • HandlerExecutionChain mappedHandler =getHandler(processedRequest)
      这里有多个handlerMapping,实际用到的是RequestMappingHandlerMapping

      this.handlerMappings ={ArrayList@14007} size =7
      0={WebMvcEndpointHandlerMapping@13963}
      1={ControllerEndpointHandlerMapping@14199}
      2={RequestMappingHandlerMapping@9698}
      3={BeanNameUrlHandlerMapping@14200}
      4= {RouterFunctionMapping@14201}
      5={SimpleUrlHandlerMapping@14202}
      6={WelcomePageHandlerMapping@14203}

  • AbstractHandlerMethodMapping
    从这个mapping 中获取RequestMappingHandlerMapping

    • HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request)

      复制代码
      protected HandlerMethod lookupHandlerMethod(String lookupPath, HttpServletRequest request) throws Exception {
      	List<Match> matches = new ArrayList<>();
      	List<T> directPathMatches = this.mappingRegistry.getMappingsByUrl(lookupPath);
      	if (directPathMatches != null) {
      		addMatchingMappings(directPathMatches, matches, request);
      	}
      	if (matches.isEmpty()) {
      		// No choice but to go through all mappings...
      		addMatchingMappings(this.mappingRegistry.getMappings().keySet(), matches, request);
      	}

这段代码有意思,先去根据url 去map中找RequestMappingInfo (封装了URL匹配相关信息) ,然后去做URL匹配

    • MappingRegistry 保存了所有的配置

      mappingLookup 保存了RequestMappingInfo 和 HandlerMethod 的映射
      MultiValueMap<String, T> urlLookup 保存了路径和方法的映射

  • 所有的配置信息封装在 RequestMappingInfo

    复制代码
      public RequestMappingInfo getMatchingCondition(HttpServletRequest request) {
      	RequestMethodsRequestCondition methods = this.methodsCondition.getMatchingCondition(request);
      	if (methods == null) {
      		return null;
      	}
      	ParamsRequestCondition params = this.paramsCondition.getMatchingCondition(request);
      	if (params == null) {
      		return null;
      	}
      	HeadersRequestCondition headers = this.headersCondition.getMatchingCondition(request);
      	if (headers == null) {
      		return null;
      	}
      	ConsumesRequestCondition consumes = this.consumesCondition.getMatchingCondition(request);
      	if (consumes == null) {
      		return null;
      	}
      	ProducesRequestCondition produces = this.producesCondition.getMatchingCondition(request);
      	if (produces == null) {
      		return null;
      	}
      	PatternsRequestCondition patterns = this.patternsCondition.getMatchingCondition(request);
      	if (patterns == null) {
      		return null;
      	}
      	RequestConditionHolder custom = this.customConditionHolder.getMatchingCondition(request);
      	if (custom == null) {
      		return null;
      	}
    
      	return new RequestMappingInfo(this.name, patterns,
      			methods, params, headers, consumes, produces, custom.getCondition());
      }

找到PatternsRequestCondition ,用来做和url做模式匹配

  • PatternsRequestCondition

    this.pathMatcher = pathMatcher != null ? pathMatcher : new AntPathMatcher();

    this.pathMatcher.match("/Task/GetTaskById/{taskid}", "/Task/GetTaskById/123")

这里实际上是spring 的AntPathMatcher 类,直接调用即可匹配带参数的URL

总结

springMVC 做URL匹配,实际上分两步,1,用url去map中找List directPathMatches

2, 遍历List directPathMatches,逐个做pathMatcher

相关推荐
Deamon Tree3 分钟前
如何保证缓存与数据库更新时候的一致性
java·数据库·缓存
rengang664 分钟前
106-Spring AI Alibaba RAG ETL Pipeline 完整案例
人工智能·spring·etl·rag·spring ai·ai应用编程
9号达人4 分钟前
认证方案的设计与思考
java·后端·面试
大G的笔记本11 分钟前
MySQL 中的 行锁(Record Lock) 和 间隙锁(Gap Lock)
java·数据库·mysql
R.lin13 分钟前
Java支付对接策略模式详细设计
java·架构·策略模式
没有bug.的程序员14 分钟前
Spring Boot 常见性能与配置优化
java·spring boot·后端·spring·动态代理
没有bug.的程序员17 分钟前
Spring Boot Actuator 监控机制解析
java·前端·spring boot·spring·源码
三次拒绝王俊凯18 分钟前
java求职学习day47
java·开发语言·学习
包饭厅咸鱼39 分钟前
autojs----2025淘宝淘金币跳一跳自动化
java·javascript·自动化
码住懒羊羊1 小时前
【C++】模板进阶 | 继承
android·java·c++