Spring boot 启动流程
框架初始化
所谓框架初始化,可以说是new 一个 SpringApplication 对象,调用其构造方法时会进行如下步骤的操作!
实际框架初始化流程,
- 配置资源加载器,默认启动,资源加载器是null (可以在启动时指定类加载器)
- 判断primarySources(servlet项目会将启动主类传入构造方法)不能为空,接下来为SpringApplication 设置primarySources
- 调用推断应用所属类型,webflux or none or servlet 应用环境检测 *具体方法在后面有详细写*
- 设置系统初始化器-》通过SpringFactoriesLoader 获取ApplicationContextInitializer 相关实现类,主要是扫描jar包下META-INF目录下的spring.factories 文件中的对应类配置
- 设置应用监听器-》也是通过SpringFactoriesLoader 获取ApplicationListener 相关实现类,
- 配置mainApplicationClass属性,也就是设置main方法所在类
探寻如上流程时,遇到一些有意思的方法
1.SpingApplication 下的 getClassLoader 方法 还会关联到 ClassUtils 中的 getDefaultClassLoader 方法,该方法应该是与双亲委派的代码很是类似。由此看出,我能看出什么呢。这个getDefaultClassLoader 方法呢,就有意思了,如果当前线程有这么一个加载器的话,就用当前线程的,没有则找到ClassUtils类的加载器,如果还是null,那么就使用systemclassloader
2.再一个就是,Throwable 类中会记录一个StackTraceElement,该元素会记录一些内容,比如当前方法所属的类名称。方法名字。。。
这个是在设置main 方法类的时候发现的。
现在深入的研究一波,getSpringFactoriesInstances 方法解析 第4步
1.先获取类加载器
2.通过SpringFactoriesLoader 的loadFactoryNames方法获 调用loadSpringFactories方法 到了set里面,保证名字的唯一
3.createSpringFactoriesInstances 方法进行对象的实例化,先通过ClassUtils.forName 获取到Class对象,再校验是否为需求的子类,通过Class对象及入参创建对应的构造方法对象,通过BeanUtils.instantiatieClass方法创建对象,并把对象放到集合中,返回对象集合。
4.对这些对象,进行一次排序根据其自定义实现的排序逻辑,或者排序器进行排序 很多都是通过order值。
loadSpringFactories流程
- 查缓存中是否包含对应数据,如果有直接返回,没有进行下一步
- 读取到META-INF目录下的spring.factories 文件
- 构造properties 对象所有的kv数据,通过遍历将这些接口及实现的名字拿到,
- 之后基于定义的接口名字为key,具体实现为value(value被声明为一个list)(接口的多个实现用“,”分割,有方法进行分割)包装到一个map中,基于这个map 取到type的具体实现类名的集合,将得到的数据放到缓存中
- 返回结果
框架启动
框架启动也就是在得到一个springApplication对象以后,进行的run方法的执行。执行会携带启动命令!
探秘框架启动流程
- 计时器开始-》new stopwatch().start()
- Headless 模式赋值
- 获取SpringApplicationRunListeners spring-boot 框架的sping-factorise 配置文件下指定了默认的启动监听器 并发送一个ApplicationStartingEvent -》 这是个扩展点
- 将启动参数包装成一个对象 也就是命令行启动时的参数会通过这一步配置到 applicationArguments 对象中
- 配置环境数据,这个方法是配置引入的入口后面进行展开。发送一个 ApplicationEnvironmentPreparedEvent标识环境信息已经准备好了处理忽略的相关内容 -》这也是个有意思的地方
- 根据已经加载了配置信息的environment对象创建打印banner对象
- 创建应用上下文对象
- 创建失败分析器
- prepareContext 准备上下文对象 下面有详解
- refreshContext(context); 对上下文进行刷新,bean的初始化都在这个方法中,后面会拓展。下一个方法在默认启动中没有具体实现代码逻辑,可能是为了实现设置的。
- 停止计时-》 打印启动哪个类 用了多长时间 的日志
- 发送started事件
- 调用runners方法 对启动加载器进行处理
- 如果以上从第4步到12步有异常或者错误发生,就会通过失败分析器,发出相关失败信息。
- 没啥事儿再发个框架正常启动的事件,然后将上下文对象
以上为框架初始化及启动相关流程,大家加油。