菜瓜:我自己去调Mvc的源码差点没给Spring的逻辑秀死。。。难受

水稻:那今天咱们看一个简单易用的SpringBoot吧

菜瓜:可以,这个我熟悉

水稻:熟悉?

菜瓜:当我没说,请开始你的表演

水稻:我没有别的意思,就是单纯的反问(手动狗头)。平时工作中用多了SpringBoot。咱们今天带着几个问题来看看它的操作吧

  1. 如何启动Spring容器
  2. 如何内嵌Tomcat容器
  3. 如何完成自动装配,就是0配置

菜瓜:你确定这是我熟悉的SpringBoot???

水稻:。。。看过来

  • 启动类点进去
  • public ConfigurableApplicationContext run(String... args) {
    StopWatch stopWatch = new StopWatch();
    stopWatch.start();
    ...
    listeners.starting();
    try {
    ...
    // ①创建Spring上下文容器对象 - 默认Servlet容器
    context = createApplicationContext();
    ...
    // ②调用refresh方法 - 回到熟悉的容器启动流程
    refreshContext(context);
    afterRefresh(context, applicationArguments);
    ...
    ...
    return context;
    }
  • ① 创建上下文对象
    • protected ConfigurableApplicationContext createApplicationContext() {
      Class<?> contextClass = this.applicationContextClass;
      if (contextClass == null) {
      try {
      switch (this.webApplicationType) {
      case SERVLET:
      contextClass = Class.forName(DEFAULT_SERVLET_WEB_CONTEXT_CLASS);
      break;
      case REACTIVE:
      contextClass = Class.forName(DEFAULT_REACTIVE_WEB_CONTEXT_CLASS);
      break;
      default:
      contextClass = Class.forName(DEFAULT_CONTEXT_CLASS);
      }
      }
      catch (ClassNotFoundException ex) {
      ...
      }
      }
      return (ConfigurableApplicationContext) BeanUtils.instantiateClass(contextClass);
      }
  • ②启动容器
    •     @Override
      public void refresh() throws BeansException, IllegalStateException {
      synchronized (this.startupShutdownMonitor) {
      prepareRefresh();
      ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory(); // Prepare the bean factory for use in this context.
      prepareBeanFactory(beanFactory); try {
      ...
      // ①springboot 内嵌tomcat容器
      onRefresh();
      ... } @Override
      protected void onRefresh() {
      super.onRefresh();
      try {
      // ②创建Servlet容器 默认tomcat
      createWebServer();
      }
      ...
      } private void createWebServer() {
      WebServer webServer = this.webServer;
      ServletContext servletContext = getServletContext();
      if (webServer == null && servletContext == null) {
      ServletWebServerFactory factory = getWebServerFactory();
      // ③看进去 回到mvc集成tomcat的场景
      this.webServer = factory.getWebServer(getSelfInitializer());
      }
      ...
      initPropertySources();
      } @Override
      public WebServer getWebServer(ServletContextInitializer... initializers) {
      Tomcat tomcat = new Tomcat();
      File baseDir = (this.baseDirectory != null) ? this.baseDirectory
      : createTempDir("tomcat");
      tomcat.setBaseDir(baseDir.getAbsolutePath());
      Connector connector = new Connector(this.protocol);
      tomcat.getService().addConnector(connector);
      customizeConnector(connector);
      tomcat.setConnector(connector);
      tomcat.getHost().setAutoDeploy(false);
      configureEngine(tomcat.getEngine());
      for (Connector additionalConnector : this.additionalTomcatConnectors) {
      tomcat.getService().addConnector(additionalConnector);
      }
      prepareContext(tomcat.getHost(), initializers);
      return getTomcatWebServer(tomcat);
      }

水稻:好了,第一步和第二步完成了

菜瓜:就这???

水稻:是不是极其简单,令人发指。重头戏是后面的自动装配

  • 回到咱们启动类的注解上
  • ...
    // 标记自身被扫描
    @SpringBootConfiguration
    // 下一步 - 自动装配入口
    @EnableAutoConfiguration
    // 扫描bean路径 - 约定是启动类所在的包:所以没事别把启动类挪走(都是泪)
    @ComponentScan(excludeFilters = {
    @Filter(type = FilterType.CUSTOM, classes = TypeExcludeFilter.class),
    @Filter(type = FilterType.CUSTOM, classes = AutoConfigurationExcludeFilter.class) })
    public @interface SpringBootApplication ->
    @Import(AutoConfigurationImportSelector.class)
    public @interface EnableAutoConfiguration -> 重要
    public class AutoConfigurationImportSelector ... {
    @Override
    public void process(AnnotationMetadata annotationMetadata,
    DeferredImportSelector deferredImportSelector) {
    ...
    // 获取以EnableAutoConfiguration命名的/META-INF/Spring.factories文件中的value去重
    AutoConfigurationEntry autoConfigurationEntry = ((AutoConfigurationImportSelector) deferredImportSelector).getAutoConfigurationEntry(getAutoConfigurationMetadata(), annotationMetadata);
    // 启动的时候断点可以看到
    this.autoConfigurationEntries.add(autoConfigurationEntry);
    for (String importClassName : autoConfigurationEntry.getConfigurations()) {
    this.entries.putIfAbsent(importClassName, annotationMetadata);
    }
    }
    }
  • AutoConfigurationImportSelector 中的process是被ConfigurationClassPostProcessor通过processConfigBeanDefinitions方法调用(调用链如下)
    1. this.processConfigBeanDefinitions(registry);
    2. parser.parse(candidates);
    3. this.parse(((AnnotatedBeanDefinition)bd).getMetadata(), holder.getBeanName());
    4. sourceClass = this.doProcessConfigurationClass(configClass, sourceClass);
    5. this.processImports(configClass, sourceClass, this.getImports(sourceClass), true);
    6. this.deferredImportSelectorHandler.handle(configClass, (DeferredImportSelector)selector);
    7. handler.processGroupImports();
    8. grouping.getImports().forEach...
    9. this.group.process(...);
    --
    搜集到需要自动装配的类,封装成BeanDefinition后续实例化,实现自动装配功能
    譬如引入WebMvcAutoConfiguration类 - webmvc功能自动集成
    org.springframework.boot.autoconfigure.web.servlet.WebMvcAutoConfiguration 

菜瓜:原来如此。你把调用链拎出来就简单了很多。自动装配就是通过SPI加载org.springframework.boot.autoconfigure包下的class,封装成BeanDefinition后交给容器加载

总结:SpringBoot只需要一行代码便能启动一个Java应用。完全解放开发者复杂的配置

  • 内嵌Servlet容器,默认tomcat
  • 启动SpringWeb容器
  • 自动装配了简单web应用需要的工具和组建

【Spring】原来SpringBoot是这样玩的的更多相关文章

  1. Spring及SpringBoot @Async配置步骤及注意事项

    前言 最近在做一个用户反馈的功能,就是当用户反馈意见或建议后服务端将意见保存然后发邮件给相关模块的开发者.考虑发邮件耗时的情况所以我想用异步的方法去执行,于是就开始研究Spring的@Async了.但 ...

  2. spring springMvc spring-boot spring-cloud分别是什么

    本文来源于:克己习礼成仁   的<spring springMvc spring-boot spring-cloud分别是什么> 什么是spring 关于spring的定义无论是从官方还是 ...

  3. Spring和SpringBoot比较,解惑区别

    1.概述: 对于Spring和SpringBoot到底有什么区别,我听到了很多答案,刚开始迈入学习SpringBoot的我当时也是一头雾水,随着经验的积累.我慢慢理解了这两个框架到底有什么区别,我相信 ...

  4. spring 或 springboot统一异常处理

    spring 或 springboot统一异常处理https://blog.csdn.net/xzmeasy/article/details/76150370 一,本文介绍spring MVC的自定义 ...

  5. 品Spring:SpringBoot和Spring到底有没有本质的不同?

    现在的Spring相关开发都是基于SpringBoot的. 最后在打包时可以把所有依赖的jar包都打进去,构成一个独立的可执行的jar包.如下图13: 使用java -jar命令就可以运行这个独立的j ...

  6. 品Spring:SpringBoot轻松取胜bean定义注册的“第一阶段”

    上一篇文章强调了bean定义注册占Spring应用的半壁江山.而且详细介绍了两个重量级的注册bean定义的类. 今天就以SpringBoot为例,来看看整个SpringBoot应用的bean定义是如何 ...

  7. 品Spring:SpringBoot发起bean定义注册的“二次攻坚战”

    上一篇文章整体非常轻松,因为在容器启动前,只注册了一个bean定义,就是SpringBoot的主类. OK,今天接着从容器的启动入手,找出剩余所有的bean定义的注册过程. 具体细节肯定会颇为复杂,同 ...

  8. Spring SpringMVC SpringBoot SpringCloud 注解整理大全

    Spring SpringMVC SpringBoot SpringCloud 注解整理 才开的博客所以放了一篇以前整理的文档,如果有需要添加修改的地方欢迎指正,我会修改的φ(๑˃∀˂๑)♪ Spri ...

  9. Spring 与 SpringBoot 的区别

    概述 Spring 与 SpringBoot 有什么区别???梳理一下 Spring 和 SpringBoot 到底有什么区别,从 Spring 和 SpringBoot 两方面入手. Spring ...

随机推荐

  1. vim的常见操作

    vim常见操作 复制 yank, y yy:复制整行 nyy/yny y^/y0:复制当前行到行头的内容 y$:复制当前到行尾的内容 yw:复制一个word nyw/ynw 复制n个word yG:复 ...

  2. Java实现 LeetCode 372 超级次方

    372. 超级次方 你的任务是计算 ab 对 1337 取模,a 是一个正整数,b 是一个非常大的正整数且会以数组形式给出. 示例 1: 输入: a = 2, b = [3] 输出: 8 示例 2: ...

  3. Java实现 LeetCode 365 水壶问题

    365. 水壶问题 有两个容量分别为 x升 和 y升 的水壶以及无限多的水.请判断能否通过使用这两个水壶,从而可以得到恰好 z升 的水? 如果可以,最后请用以上水壶中的一或两个来盛放取得的 z升 水. ...

  4. Java实现 蓝桥杯VIP 算法提高 3-3求圆面积表面积体积

    算法提高 3-3求圆面积表面积体积 时间限制:1.0s 内存限制:256.0MB 问题描述 接受用户输⼊的数值,输出以该值为半径的(1)圆面积,(2)球体表面积,(3)球体体积.pi 取值3.1415 ...

  5. Java实现 蓝桥杯VIP 算法提高 夺宝奇兵

    算法提高 夺宝奇兵 时间限制:1.0s 内存限制:512.0MB [题目描述] 在一座山上,有很多很多珠宝,它们散落在山底通往山顶的每条道路上,不同道路上的珠宝的数目也各不相同.下图为一张藏宝地图: ...

  6. Java实现 LeetCode 61 旋转链表

    61. 旋转链表 给定一个链表,旋转链表,将链表每个节点向右移动 k 个位置,其中 k 是非负数. 示例 1: 输入: 1->2->3->4->5->NULL, k = ...

  7. shell中文本内容多行变一行的技巧

    在linux下有时可能需要将多行的值转成一行.其实现的方法有很多种.笔者将自己曾经用过的方法在些分享. 如有一文本文件5201351.txt,文本的内容如下: 现我们可以通过如下方法将文本内容转成一行 ...

  8. 关于nginx的源码安装方式

    Nginx(engine x)是一款是由俄罗斯的程序设计师Igor Sysoev所开发高性能的 Web和 反向代理 服务器, 也是一个 IMAP/POP3/SMTP 代理服务器.在高连接并发的情况下, ...

  9. 从程序员到项目主管再到项目总监,一个IT从业者三个职业生涯阶段的工作生活日常

    这是王不留的第 8 篇原创文章 前段时间写过<王不留的十多年工作和生活的流水帐>,在知乎.简书,还有不少微信的朋友私信问我每天四点钟是如何做到的?你现在的作息时间是怎么安排的? 于是,我将 ...

  10. 手把手带你入门numpy,从此数据处理不再慌【四】

    本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是numpy专题的第四篇文章,numpy中的数组重塑与三元表达式. 首先我们来看数组重塑,所谓的重塑本质上就是改变数组的shape.在保 ...