为了账号安全,请及时绑定邮箱和手机立即绑定

RequestMappingHandlerMapping 详解

标签:
Java

我们先理简单梳理一个关系

关系梳理

  1. spring ioc 是spring的核心,用来管理spring bean的生命周期

  2. MVC 是一种使用 MVC(Model View Controller 模型-视图-控制器)设计创建 Web 应用程序的模式

  3. 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 并且按照顺序使用

private void initHandlerMappings(ApplicationContext context) {
       this.handlerMappings = null;
       if (this.detectAllHandlerMappings) {
           // Find all HandlerMappings in the ApplicationContext, including ancestor contexts.
           Map<String, HandlerMapping> matchingBeans =
                   BeanFactoryUtils.beansOfTypeIncludingAncestors(context, HandlerMapping.class, true, false);
           if (!matchingBeans.isEmpty()) {
               this.handlerMappings = new ArrayList<>(matchingBeans.values());
               // We keep HandlerMappings in sorted order.
               AnnotationAwareOrderComparator.sort(this.handlerMappings);
           }
       }
}

RequestMappingHandlerMapping 加载过程

  1. RequestMappingHandlerMapping 实现了接口InitializingBean,在bean加载完成后会自动调用afterPropertiesSet方法,在此方法中调用了initHandlerMethods()来实现初始化

  2. 遍历所有bean,如果bean实现带有注解@Controller或者@RequestMapping 则进一步调用detectHandlerMethods处理,处理逻辑大致就是根据@RequestMapping配置的信息,构建RequestMappingInfo,然后注册到MappingRegistry中

protected void initHandlerMethods() {
       String[] beanNames = (this.detectHandlerMethodsInAncestorContexts ?
               BeanFactoryUtils.beanNamesForTypeIncludingAncestors(obtainApplicationContext(), Object.class) :
               obtainApplicationContext().getBeanNamesForType(Object.class));
       for (String beanName : beanNames) {
           if (!beanName.startsWith(SCOPED_TARGET_NAME_PREFIX)) {
               Class<?> beanType = null;
               beanType = obtainApplicationContext().getType(beanName);
               if (beanType != null && isHandler(beanType)) {
                   detectHandlerMethods(beanName);
               }
           }
       }
       handlerMethodsInitialized(getHandlerMethods());
   }

protected void detectHandlerMethods(final Object handler) {
       Class<?> handlerType = (handler instanceof String ?
               obtainApplicationContext().getType((String) handler) : handler.getClass());
       if (handlerType != null) {
           final Class<?> userType = ClassUtils.getUserClass(handlerType);
           Map<Method, T> methods = MethodIntrospector.selectMethods(userType,
                   (MethodIntrospector.MetadataLookup<T>) method -> {
                       try {
                           return getMappingForMethod(method, userType);
                       }
                       catch (Throwable ex) {
                           throw new IllegalStateException("Invalid mapping on handler class [" +
                                   userType.getName() + "]: " + method, ex);
                       }
                   });
           methods.forEach((method, mapping) -> {
               Method invocableMethod = AopUtils.selectInvocableMethod(method, userType);
               registerHandlerMethod(handler, invocableMethod, mapping);
           });
       }
   }

RequestMappingHandlerMapping 解析过程

  1. 在DispatcherServlet中,根据请求对象调用getHander方法获取HandlerExecutionChain对象

  2. 在getHander方法中也是遍历上面默认加载的三个HandlerMapping,当然第一个就是RequestMappingHandlerMapping对象,调用其getHandler方法,根据请求path,找到一个最为匹配的HandlerMethod来处理请求

protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
       if (this.handlerMappings != null) {
           for (HandlerMapping hm : this.handlerMappings) {
               if (logger.isTraceEnabled()) {
                   logger.trace(
                           "Testing handler map [" + hm + "] in DispatcherServlet with name '" + getServletName() + "'");
               }
               HandlerExecutionChain handler = hm.getHandler(request);
               if (handler != null) {
                   return handler;
               }
           }
       }
       return null;
   }
  1. 根据请求路径获取HandlerInterceptor,然后和上面获得的HandlerMethod一起构成HandlerExecutionChain返回给DispatcherServlet

DispatcherServlet得到HandlerExecutionChain也就获得了处理此次请求所需的Handler【即我们熟悉的Controller和对应的Action】,后续将会选择合适HandlerAdapter来执行对应的Handler,获取返回值,再根据返回值类型,进一步觉决定用什么方式展示给用户,下一遍将开启HandlerAdapter的讲解…….

往期spring系列文章也可参考  spring 系列


点击查看更多内容
1人点赞

若觉得本文不错,就分享一下吧!

评论

作者其他优质文章

正在加载中
JAVA开发工程师
手记
粉丝
5
获赞与收藏
27

关注作者,订阅最新文章

阅读免费教程

感谢您的支持,我会继续努力的~
扫码打赏,你说多少就多少
赞赏金额会直接到老师账户
支付方式
打开微信扫一扫,即可进行扫码打赏哦
今天注册有机会得

100积分直接送

付费专栏免费学

大额优惠券免费领

立即参与 放弃机会
意见反馈 帮助中心 APP下载
官方微信

举报

0/150
提交
取消