SpringMVC 遗漏补充

简介: 说到SpringMVC,一般都说Spring将请求与方法进行了映射,所以每次请求都能找到对应的方法,这次我想找到代码层面上是如何处理的,而不是泛泛而谈。我们在springmvc启动的时候说到,由于DispatcherServlet 实现了Servlet,因此会默认实现其init( )方法.

说到SpringMVC,一般都说Spring将请求与方法进行了映射,所以每次请求都能找到对应的方法,这次我想找到代码层面上是如何处理的,而不是泛泛而谈。我们在springmvc启动的时候说到,由于DispatcherServlet 实现了Servlet,因此会默认实现其init方法,在init方法中找到了我比较感兴趣的两个函数
(1) initHandlerMappings(context)
(2) initHandlerAdapters(context)

这两个函数的内部实现不贴代码了,就是将一些默认的类经常初始化到容器,供请求过来的时候进行调用。ok,这是我前几次看完代码得到的感悟,当时始终没有解决我的一个疑问,到底在什么时候将URL与方法进行了映射?通过Debug跟踪请求也是莫名其妙的Map里面就存在了映射关系,实在没有办法只能反向查找,终于在 AbstractHandlerMethodMapping 这个类中找到了答案。

public abstract class AbstractHandlerMethodMapping<T> 
         extends AbstractHandlerMapping implements InitializingBean

AbstractHandlerMethodMapping 是SpringMVC默认的一个Handler,由于其实现了InitializingBean的接口,那么在容器启动的时候就会调用:

@Override
public void afterPropertiesSet() {
    initHandlerMethods();
}

step1:

 BeanFactoryUtils.beanNamesForTypeIncludingAncestors(getApplicationContext(), Object.class)
 扫描引用上下文,获取所有的Bean

step2:

  Class<?> beanType = getApplicationContext().getType(beanName);
  
   protected boolean isHandler(Class<?> beanType) {
    return (AnnotatedElementUtils.hasAnnotation(beanType, Controller.class) ||
           AnnotatedElementUtils.hasAnnotation(beanType, RequestMapping.class));
}

遍历所有的Bean,筛选拥有注解Controller 或者 RequestMapping的类

step3:

  进入最主要的函数体:detectHandlerMethods
protected void detectHandlerMethods(final Object handler) {
    
    // 获取handler类型
    Class<?> handlerType = (handler instanceof String ? getApplicationContext().getType((String) handler) : handler.getClass());
    final Class<?> userType = ClassUtils.getUserClass(handlerType);

        //  从实体类中获取方法(一般都是controller类)  
        //  这边有兴趣的话可以关注一下selectMethods的实现,应该会在工作中会有涉及到
        //  使用反射的方式获所有的方法
    Map<Method, T> methods = MethodIntrospector.selectMethods(userType,
        new MethodIntrospector.MetadataLookup<T>() {
        @Override
        public T inspect(Method method) {
            return getMappingForMethod(method, userType);
        }
    });

        for (Map.Entry<Method, T> entry : methods.entrySet()) {
        Method invocableMethod = AopUtils.selectInvocableMethod(entry.getKey(), userType);
        T mapping = entry.getValue();
        // 将其映射关系放入到一个对象里面
        registerHandlerMethod(handler, invocableMethod, mapping);
    }
}

大致就是这样的一个过程,感觉像是记录的一个流水账,哈哈

补充:

 HandlerMappings在DispatcherServlet的主要作用就是为请求匹配对应的Controller,建议一个映射关系,根据请求查找Handler,Interceptor。HandlerMappings将请求传递到HandlerExecutionChain中,HandlerExecutionChain包含了一个能处理该请求的处理器,已经需要的拦截器。
 DispatcherServlet在没有配置处理器的情况下,会默认的从 DispatcherServlet.properties 配置文件中初始化类:BeanNameUrlHandlerMapping/DefaultAnnotationHandlerMapping

 AbstractHandlerMapping是接口HandlerMappings的抽象实现,是所有的HandlerMapping实现类的父类,其作用是初始化拦截器,将拦截器放入到adaptedInterceptors中,在请求过来的时候,调用getHandlerExecutionChain方法,会遍历这个adaptedInterceptors集合,选择匹配的拦截器。
目录
相关文章
组合计数及补充
组合计数及补充
73 0
|
机器学习/深度学习
【知识补充】
【知识补充】
50 0
|
5月前
codereview开发问题之CodeReview中如何判断注释问题如何解决
codereview开发问题之CodeReview中如何判断注释问题如何解决
|
5月前
|
存储 Java
软件开发常用之SpringBoot文件上传接口编写(中),一本书,代码大全(里面有很多代码重构的方法),屎山代码的原因是不断追加逻辑,在错误代码上堆积新的功能,在写完逻辑之后去思考一下,逻辑合理不
软件开发常用之SpringBoot文件上传接口编写(中),一本书,代码大全(里面有很多代码重构的方法),屎山代码的原因是不断追加逻辑,在错误代码上堆积新的功能,在写完逻辑之后去思考一下,逻辑合理不
|
7月前
|
Java
提高代码质量的秘诀:类、方法、字段和包注释
提高代码质量的秘诀:类、方法、字段和包注释
68 0
|
NoSQL 关系型数据库 MySQL
一日一技:如何正确为历史遗留代码补充单元测试?
一日一技:如何正确为历史遗留代码补充单元测试?
102 0
|
XML JSON Java
【Spring专题】「原理系列」SpringMVC的运行工作原理(补充修订)
【Spring专题】「原理系列」SpringMVC的运行工作原理(补充修订)
118 0
【Spring专题】「原理系列」SpringMVC的运行工作原理(补充修订)
|
Kubernetes 容器
k8s补充
k8s补充
|
JSON 数据可视化 JavaScript
开发改了接口,经常忘通知测试,有什么好的解决方案吗?
不知道大家有没有同感,做接口测试麻烦的不是测试本身,而是接口它会变,更麻烦的不是接口变了,而是它变了而你不知道。等到你测完,开发才悠悠跟你说——“那个接口我改了点东西,你再看一眼哈”。
开发改了接口,经常忘通知测试,有什么好的解决方案吗?
|
Java 容器 Spring
【Spring注解驱动开发】如何实现方法、构造器位置的自动装配?我这样回答让面试官很满意!
在 冰河技术 微信公众号前面的文章中,我们介绍了如何使用注解来自动装配Spring组件。之前将的都是在来的字段上添加注解,那有没有什么方法可以实现方法、构造器位置的自动装配吗?今天我们就一起来探讨下如何实现方法、构造器位置的自动装配。
112 0