美文网首页
十 SpringApplication.run() 做了些啥?

十 SpringApplication.run() 做了些啥?

作者: cf6bfeab5260 | 来源:发表于2019-04-08 11:01 被阅读0次

直接来看一下源码:

public ConfigurableApplicationContext run(String... args) {
        StopWatch stopWatch = new StopWatch();
        stopWatch.start();
        ConfigurableApplicationContext context = null;
        Collection<SpringBootExceptionReporter> exceptionReporters = new ArrayList<>();
        configureHeadlessProperty();
      //获取所有的listener,并启动
        SpringApplicationRunListeners listeners = getRunListeners(args);
        listeners.starting();
        try {
            ApplicationArguments applicationArguments = new DefaultApplicationArguments(
                    args);
        //环境准备
            ConfigurableEnvironment environment = prepareEnvironment(listeners,
                    applicationArguments);
        //设置需要忽略的bean
            configureIgnoreBeanInfo(environment);
        //打印banner
            Banner printedBanner = printBanner(environment);
       //初始化context
            context = createApplicationContext();
            exceptionReporters = getSpringFactoriesInstances(
                    SpringBootExceptionReporter.class,
                    new Class[] { ConfigurableApplicationContext.class }, context);
            prepareContext(context, environment, listeners, applicationArguments,
                    printedBanner);
            refreshContext(context);
            afterRefresh(context, applicationArguments);
            stopWatch.stop();
            if (this.logStartupInfo) {
                new StartupInfoLogger(this.mainApplicationClass)
                        .logStarted(getApplicationLog(), stopWatch);
            }
            listeners.started(context);
            callRunners(context, applicationArguments);
        }
        catch (Throwable ex) {
            handleRunFailure(context, ex, exceptionReporters, listeners);
            throw new IllegalStateException(ex);
        }

        try {
            listeners.running(context);
        }
        catch (Throwable ex) {
            handleRunFailure(context, ex, exceptionReporters, null);
            throw new IllegalStateException(ex);
        }
        return context;
    }

详细分析:

  1. 如果我们使用的是SpringApplication的静态run方法,那么,这个方法里面首先要创建一个SpringApplication对象实例,然后调用这个创建好的SpringApplication的实例方法。在SpringApplication实例初始化的时候,它会提前做几件事情:
  • 根据classpath里面是否存在某个特征类(org.springframework.web.context.ConfigurableWebApplicationContext)来决定是否应该创建一个为Web应用使用的ApplicationContext类型。
  • 使用SpringFactoriesLoader在应用的classpath中查找并加载所有可用的ApplicationContextInitializer。
  • 使用SpringFactoriesLoader在应用的classpath中查找并加载所有可用的ApplicationListener。
  • 推断并设置main方法的定义类。
  1. SpringApplication实例初始化完成并且完成设置后,就开始执行run方法的逻辑了,方法执行伊始,首先遍历执行所有通过SpringFactoriesLoader可以查找到并加载的SpringApplicationRunListener。调用它们的started()方法,告诉这些SpringApplicationRunListener,“嘿,SpringBoot应用要开始执行咯!”。
  2. 创建并配置当前Spring Boot应用将要使用的Environment(包括配置要使用的PropertySource以及Profile)。
  3. 遍历调用所有SpringApplicationRunListener的environmentPrepared()的方法,告诉他们:“当前SpringBoot应用使用的Environment准备好了咯!”。
  4. 如果SpringApplication的showBanner属性被设置为true,则打印banner。banner就是启动的时候控制台打印的那个很丑的springboot图标。
  5. 根据用户是否明确设置了applicationContextClass类型以及初始化阶段的推断结果,决定该为当前SpringBoot应用创建什么类型的ApplicationContext并创建完成,然后根据条件决定是否添加ShutdownHook,决定是否使用自定义的BeanNameGenerator,决定是否使用自定义的ResourceLoader,当然,最重要的,将之前准备好的Environment设置给创建好的ApplicationContext使用。
  6. ApplicationContext创建好之后,SpringApplication会再次借助Spring-FactoriesLoader,查找并加载classpath中所有可用的ApplicationContext-Initializer,然后遍历调用这些ApplicationContextInitializer的initialize(applicationContext)方法来对已经创建好的ApplicationContext进行进一步的处理。
  7. 遍历调用所有SpringApplicationRunListener的contextPrepared()方法。
  8. 最核心的一步,将之前通过@EnableAutoConfiguration获取的所有配置以及其他形式的IoC容器配置加载到已经准备完毕的ApplicationContext。
  9. 遍历调用所有SpringApplicationRunListener的contextLoaded()方法。
  10. 调用ApplicationContext的refresh()方法,完成IoC容器可用的最后一道工序。
  11. 查找当前ApplicationContext中是否注册有CommandLineRunner,如果有,则遍历执行它们。
  12. 正常情况下,遍历执行SpringApplicationRunListener的finished()方法、(如果整个过程出现异常,则依然调用所有SpringApplicationRunListener的finished()方法,只不过这种情况下会将异常信息一并传入处理)
    流程如下:


    0.jpg

参考链接:https://www.cnblogs.com/zheting/p/6707035.html

相关文章

  • 十 SpringApplication.run() 做了些啥?

    直接来看一下源码: 详细分析: 如果我们使用的是SpringApplication的静态run方法,那么,这个方法...

  • 2018做了些啥?

    一晃2018就到最后几天,日子真是过得恍恍惚惚,回顾这一年,好像也是一样平平淡淡过来了。 稍稍有些欣慰是工作上有看...

  • SpringApplication.run 到底做了什么?

    用过 SpringBoot 的同学都知道,其程序的启动类是在一个main方法中调用SpringApplicatio...

  • SpringBoot启动

    SpringApplication.run一共做了两件事,一件是创建SpringApplication对象,在该对...

  • 加班我都做了些啥

    下午去调查了,回来写了一下调查报告,很快就到了下班时间。 下班回到家,我炒了鸡蛋,青菜加面条,然后先去药店买药,根...

  • springboot启动原理

    SpringApplication.run() 做了两件事情。 1. 判断当前项目类型,web类型。 2. 初始化...

  • springboot初始化过程(源代码注释)

    springboot启动入口类 调用了SpringApplication.run方法 对一些基本的配置进行初始化,...

  • 2016年,我都做了些啥?

    “一年的工作、生活、学习正式新的启程。今年希望自己:坚持运动,游泳、跑步、骑车等等,塑身,学跳舞、网球;学编程;继...

  • 我这半年都做了些啥?

    大家看我是不是瘦了,是的,真的瘦了,瘦了6斤,虽然不多,但我自豪,为什么呢? 因为我这半年就做了一件事,运动,每天...

  • https中的s做了些啥

    一、https是通过HTTP协议和SSL/TLS协议共同完成的 二、用到了对称加密和非对称加密(内容是对称加密,对...

网友评论

      本文标题:十 SpringApplication.run() 做了些啥?

      本文链接:https://www.haomeiwen.com/subject/gqtuiqtx.html