SpringBoot RUN方法执行流程

SpringBoot RUN方法执行流程

1、查看main方法

@SpringBootApplication // 能够扫描Spring组件并自动配置Spring Boot
public class  SpringbootApplication {
    public static void main(String[] args) {
        SpringApplication.run(SpringbootApplication.class, args);
    }
}

2、点进run方法

public static ConfigurableApplicationContext run(Class<?> primarySource, String... args) { // 调用重载方法
        return run(new Class[]{primarySource}, args);
    }

    public static ConfigurableApplicationContext run(Class<?>[] primarySources, String[] args) { // 两件事:1.初始化SpringApplication 2.执行run方法
        return (new SpringApplication(primarySources)).run(args);
    }
  • 初始化SpringApplication对象
  • 执行run方法

3、初始化SpringApplication对象

public SpringApplication(Class<?>... primarySources) {
        this((ResourceLoader)null, primarySources);
    }

    public SpringApplication(ResourceLoader resourceLoader, Class<?>... primarySources) {
        this.sources = new LinkedHashSet();
        this.bannerMode = Mode.CONSOLE;
        this.logStartupInfo = true;
        this.addCommandLineProperties = true;
        this.addConversionService = true;
        this.headless = true;
        this.registerShutdownHook = true;
        this.additionalProfiles = Collections.emptySet();
        this.isCustomEnvironment = false;
        this.lazyInitialization = false;
        this.applicationContextFactory = ApplicationContextFactory.DEFAULT;
        this.applicationStartup = ApplicationStartup.DEFAULT;
        // 设置资源加载器
        this.resourceLoader = resourceLoader;
        // 断言加载资源类不能为null
        Assert.notNull(primarySources, "PrimarySources must not be null");
        // 将primarySources数组转换为List,最后放到LinkedHashSet集合中
        this.primarySources = new LinkedHashSet(Arrays.asList(primarySources));
        //【1.1 推断应用类型,后面会根据类型初始化对应的环境。常用的一般都是servlet环境 】
        this.webApplicationType = WebApplicationType.deduceFromClasspath();
        this.bootstrapRegistryInitializers = this.getBootstrapRegistryInitializersFromSpringFactories();
				
// 【1.2 初始化classpath下 META-INF/spring.factories中已配置的 ApplicationContextInitializer 】       
      this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));
// 【1.3 初始化classpath下所有已配置的 ApplicationListener 】
        this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));
// 【1.4 根据调用栈,推断出 main 方法的类名 】
        this.mainApplicationClass = this.deduceMainApplicationClass();
    }

4、 run(args)源码剖析

public ConfigurableApplicationContext run(String... args) {
        //记录程序运行时间
        StopWatch stopWatch = new StopWatch();
        stopWatch.start();
  	    // 注解bootstrap 初始化器到BootStrapContext上下文中
        DefaultBootstrapContext bootstrapContext = this.createBootstrapContext();
        ConfigurableApplicationContext context = null;
  			// 设置 java.awt.headless属性
        this.configureHeadlessProperty();
  			//从META-INF/spring.factories中获取监听器 `SpringApplicationRunListener`
        //1、获取并启动监听器
        SpringApplicationRunListeners listeners = this.getRunListeners(args);
  			// 循环遍历启动监听器
        listeners.starting(bootstrapContext, this.mainApplicationClass);

        try {
          
            // 封装参数对象
            ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);
            //2、构造应用上下文环境,看构建的是SERVLET、REACTIVE、还是普通环境
            ConfigurableEnvironment environment = this.prepareEnvironment(listeners, bootstrapContext, applicationArguments);
            //处理需要忽略的Bean
            this.configureIgnoreBeanInfo(environment);
            //打印banner
            Banner printedBanner = this.printBanner(environment);
            //3、初始化应用上下文
            context = this.createApplicationContext();
            context.setApplicationStartup(this.applicationStartup);
          //4、刷新应用上下文前的准备阶段
            this.prepareContext(bootstrapContext, context, environment, listeners, applicationArguments, printedBanner);
          //5、刷新应用上下文
            this.refreshContext(context);
          //刷新应用上下文后的扩展接口
            this.afterRefresh(context, applicationArguments);
          //时间记录停止
            stopWatch.stop();
            if (this.logStartupInfo) {
                (new StartupInfoLogger(this.mainApplicationClass)).logStarted(this.getApplicationLog(), stopWatch);
            }
						//发布容器启动完成事件
            listeners.started(context);
          	// 将ApplicationRunner、CommandLineRunner注入到list里面并排序,然后循环调用Runner的run方法
            this.callRunners(context, applicationArguments);
        } catch (Throwable var10) {
            this.handleRunFailure(context, var10, listeners);
            throw new IllegalStateException(var10);
        }

        try {
            listeners.running(context);
            return context;
        } catch (Throwable var9) {
            this.handleRunFailure(context, var9, (SpringApplicationRunListeners)null);
            throw new IllegalStateException(var9);
        }
    }

以后就是主要分六步

  • 第1步:创建并启动监听器listener

  • 第2步:构建上下文环境,看是创建看构建的是SERVLET、REACTIVE、还是普通环境ApplicationEnvironment

  • 第3步:初始化应用上下文

  • 第4步:刷新应用上下文前的准备阶段,

  • 第5步:刷新上下文并且注册了一个钩子(JVM关闭就会关闭这个上下文)

  • 第6步:刷新应用上下文后的扩展接口,afterRefresh这个是一个空实现的方法

Spring Boot的run启动流程大致如下: 1. 创建SpringApplication对象:SpringApplicationSpring Boot的核心类,用于启动应用程序。在创建SpringApplication对象时,需要传入一个主配置类。 2. 解析应用程序的配置:SpringApplication会解析主配置类上的注解,并加载配置信息,包括应用程序的配置文件、默认配置等。 3. 创建ApplicationContext:SpringApplication根据配置信息创建一个ApplicationContext对象,用于管理和加载应用程序的Bean。 4. 预处理ApplicationContext:在ApplicationContext创建之前,可以注册一些监听器和处理器,对ApplicationContext进行预处理。 5. 加载ApplicationContext:SpringApplication会根据配置信息,加载所有的Bean定义,并将它们注册到ApplicationContext中。 6. 刷新ApplicationContext:刷新ApplicationContext会触发各种事件和监听器,完成Bean的实例化、依赖注入等操作。 7. 执行CommandLineRunner:如果应用程序实现了CommandLineRunner接口,Spring Boot会在ApplicationContext刷新完成后,调用其run方法。 8. 启动Web服务器:如果应用程序是一个Web应用程序,Spring Boot会根据配置信息启动内嵌的Web服务器(如Tomcat、Jetty等)。 9. 应用程序启动完成:至此,Spring Boot的run启动流程完成,应用程序已经成功启动。 需要注意的是,以上只是Spring Boot启动流程的一个简单描述,实际流程可能会因为应用程序的复杂性和定制化配置而有所变化。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值