SpringApplication.run(JvmStudyApplication.class, args);
当我们这样执行一个run方法的时候,首先会构造一个springApplication,然后再去启动run方法
构造SpringApplication的过程:
1,设置webApplicationType,判断应用类型,这是springboot的一个成员变量,通过判断jar包里有没有javax.servlet.Servlet接口
和org.springframework.web.context.ConfigurableWebApplicationContext类,如果都有,那么就是servlet类型的
2,setInitializers:设置启动器,通过getSpringFactoriesInstances方法,把ApplicationContextInitializer的class对象传进去,然后寻找所有jar文件里的META-INF下面的spring.factories,从这个文件读取信息,里面是键值对,键是一个接口的全限定名,值是实现它的所有实现类,值有多个,用了一个MultiValueMap存起来,来自于不通的spring.factories会有相同的key,MultiValueMap是多只得,所以会合并,然后把类加载器作为key,MultiValueMap对象作为值存到缓存中,然后再找到key为ApplicationContextInitializer的所有值,通过反射的方式创建出它们的实例
3,setListeners:设置监听器,创建ApplicationListener接口所有的实现类,和2中步骤一样,只是这次是从缓存中读取
4,给mainApplicationClass赋值,通过new RuntimeException.getStackTrace获取main方法所在的类,也就是启动类,赋值给mainApplicationClass。
run方法的过程:
1,秒表计时器开始
2,获取SpringApplicationRunListeners,调用监听器的Starting方法,表明run方法首次启动
3,构造一个应用参数对象,配置环境,根据webApplicationTYpe,如果是servlet,就构造一个StandaredServletEnvironment,配置好profile,然后调用environmentPrepared方法,表示环境准备好了,但是还没创建ApplicationContext
4,打印banner
5,创建ConfigurableApplicationContext 根据我们的应用类型是标准的servlet类型,所以创建一个org.springframework.boot.web.servlet.context.AnnotationConfigServletWebServerApplicationContext实例
6,创建SpringBootExceptionReporter接口实现类的实例
7,准备上下文,把之前获取到的环境设置到上下文中去,调用监听器的contextprepared方法,表示ApplicationContext创建好和准备好了,但是还没被加载源
8,打印profile相关的信息,加载源(也就是启动类),调用监听器的contextLoaded方法,表明上下文已经被加载了,但是在被刷新之前
9,刷新上下文
10,打印启动时间,然后调用监听器的started方法
11,调用callRunners方法
12,调用监听器的running方法。