SpringMvc的原理深度剖析及源码解读

一、springmvc启动加载流程

1、引入spring-web.jar包时,在这个包的META-INF/services/javax.servlet.ServletContainerInitializer文件中定义的加载类SpringServletContainerInitializer,提供给springmvc实现初始化的操作。

2、在SpringServletContainerInitializer类中,通过@HandlesTypes来引入WebApplicationInitializer。

3、自己自定义的启动类只要继承了WebApplicationInitializer类并实现了onStartUp方法,那么servlet容器在启动的时候就会拿到自己自定义的启动类,并开始启动。

**启动原理:**servlet容器在初始化的时候会加载到SpringServletContainerInitializer类,然后这个类中通过@HandlesTypes来引入WebApplicationInitializer子类,然后回调传递ServletContext上下文,此处使用的是模板方法。

springmvc是如何来替代web.xml的?核心就是使用的SpringServletContainerInitializer类。

二、拦截器与过滤器区别

相同点:

拦截器和过滤器都是基于Aop实现,能够对请求执行之前和之后实现拦截。

不同点:

过滤器是基于Servlet容器实现,对Web请求之前和之后实现拦截

拦截器不需要依赖于Servlet、不仅可以实现Web请求进行拦截,还有对其他方法拦截等。

过滤器比拦截器先执行,拦截器封装的方法比过滤器拦截使用起来更加简单。

应用场景:

拦截器:权限控制、日志打印、参数验证、会话信息。

过滤器:编码转换、跨域解决、xss攻击。

三、DispatcherServlet底层源码分析

1.执行doDispatch;

2.调用getHandler方法获取请求目标的方法 也就是 请求url映射路径对应的控制层具体的方法;

handlerMappings的作用查找控制器位置,比如xml和注解方式;

3.调用getHandlerAdapter获取控制层适配器 RequestMappingHandlerAdapter;

4.执行拦截器前置方法 preHandle() 如果返回为true的话;

5.执行实际请求目标方法 返回modeAndView对象;

6.执行拦截器PostHandle()方法;

7.设置渲染视图层内容;

8.执行拦截器afterCompletion方法;

四、SpringMVC控制对象初始化流程

复制代码
1、HttpServletBean类的init() 方法

2、FrameworkServlet类的initServletBean() 方法 →  initWebApplicationContext()

3、DispatcherServlet类的onRefresh() 方法 →  initStrategies()方法
java 复制代码
protected void initStrategies(ApplicationContext context) {

            initMultipartResolver(context);//初始化上传文件解析器(或者是多部分请求解析器)
           
            initLocaleResolver(context);//初始化本地化解析器
            
            initThemeResolver(context);//初始化主题解析器
                 
            initHandlerMappings(context);//初始化处理器映射器
                 
            initHandlerAdapters(context);//初始化处理器适配器
                 
            initHandlerExceptionResolvers(context);//初始化处理器异常解析器
                 
            initRequestToViewNameTranslator(context);//初始化请求到视图名翻译器
                 
            initViewResolvers(context);//初始化视图解析器
                 
            initFlashMapManager(context);//初始化重定向数据管理器
}

五、SpringMVC中的适配器

**适配器模式:**将一个系统的接口转换成另外一种形式,从而使原来不能直接调用的接口变得可以调用。使用适配器可以针对不容的handler类型找到不同的适配器来实现执行目标方法。

SpringMVC中有以下几种适配器:

1、继承Controller方式所使用的适配器:SimpleControllerHandlerAdapter

2、 HTTP请求处理器适配器:HttpRequestHandlerAdapter

3、注解方式(@Controller)的处理器适配器:RequestMappingHandlerAdapter

适配器模式应用场景

  1. Mybatis多种日志框架的整合(logback、log4j)

  2. SpringMVC适配器模式

  3. 新老版本的兼容问题

适配器模式源码分析

java 复制代码
1.使用getHandlerAdapter获取对应的hanlder的具体HandlerAdapter

2.HandlerAdapter接口有如下的子 c处理请求适配器

    2.1继承Controller方式所使用的适配器:SimpleControllerHandlerAdapter

    2.2 HTTP请求处理器适配器:HttpRequestHandlerAdapter 
    
    2.3注解方式(@Controller)的处理器适配器:RequestMappingHandlerAdapter


如果不采用适配器的话 :
If(hanlder instanceof Controller){
 // 执行Controller适配器
}

If(hanlder instanceof  HttpControler){
 // 执行我们的HttpController
}

If(hanlder instanceof  ServletControler){
 // 执行我们的HttpController
}

If(hanlder instanceof  AnnotationControler){
 // 执行我们的AnnotationController
}

六、SpringMvc异步实现方式

异步支持需要注意的两个地方

* 1、config类中开启异步@EnableAsync

* 2、初始化类中 dynamic.setAsyncSupported(true);//开启springmvc异步

1、第一种方式:直接在接口上添加@Async注解来实现

java 复制代码
@Async
public String getMember(){
    try {
        System.out.println("2、业务处理开始,线程名称"+Thread.currentThread().getContextClassLoader());
        Thread.sleep(3000);
        System.out.println("3、业务处理完毕,线程名称"+Thread.currentThread().getContextClassLoader());
    }catch (Exception e){
        e.printStackTrace();
    }
    return "member";
}

2、第二种方式:使用Callable类的方式

java 复制代码
/**
 * 异步支持需要注意的两个地方
 * 1、config类中开启异步@EnableAsync
 * 2、初始化类中 dynamic.setAsyncSupported(true);//开启springmvc异步
 * Callable这种方式也是需要等业务线程处理完毕后,才会返回给客户端,为了避免等待时间,可以在客户端写一个定时器来查询处理结果
 */
@RequestMapping("/asyncMember")
@ResponseBody
public Callable<String> asyncMember(){
    Callable<String> callable = new Callable<String>() {
        public String call() throws Exception {
            //耗时的时间都可以放在这个里面去处理
            String member = memberService.getMember();
            return member;
        }
    };
    return callable;
}
相关推荐
期待のcode4 小时前
SpringAOP
java·开发语言·spring
算家计算5 小时前
5B 参数,消费级显卡可部署:Wan2.2-TI2V-5B 本地部署教程,9分钟跑出电影级大片!
人工智能·开源·aigc
dylan_QAQ5 小时前
【附录】相对于BeanFactory ,ApplicationContext 做了哪些企业化的增强?
后端·spring
he___H6 小时前
黑马SpringAI项目-聊天机器人
spring·springai
说私域6 小时前
基于梅特卡夫定律的开源链动2+1模式AI智能名片S2B2C商城小程序价值重构研究
人工智能·小程序·开源
写bug写bug6 小时前
搞懂Spring任务执行器和调度器模型
java·后端·spring
tanxiaomi6 小时前
✨ 基于 JsonSerialize 实现接口返回数据的智能枚举转换(优雅告别前端硬编码!)
java·前端·spring·spring cloud·mybatis
dylan_QAQ8 小时前
【附录】为什么说 Spring 中 BeanFactory的是延迟加载 和轻量级的?有什么证据?
后端·spring
许苑向上8 小时前
【Spring IoC 核心实现类详解:DefaultListableBeanFactory】
spring boot·spring·ioc