springboot执行流程
构造方法初始化,创建一个新的实例,这个应用程序的上下文要从指定的来源加载bean
public SpringApplication(ResourceLoaderresourceLoader,Class<?>...primarySources){
this.sources=newLinkedHashSet();
this.bannerMode=Mode.CONSOLE;
this.logStartupInfo=true;
this.addCommandLineProperties=true;
this.headless=true;
this.registerShutdownHook=true;
this.additionalProfiles=newHashSet();
初始化资源加载器,默认为null
this.resourceLoader=resourceLoader;
断言要加载的资源类不能为null否则报错
Assert.notNull(primarySources,"Primary Sources must not be null");
初始化主要加载资源并去重
this.primarySources=newLinkedHashSet(Arrays.asList(primarySources));
判断当前应用程序的类型 NONE/SERVLET/REACTIVE
this.webApplicationType=this.deduceWebApplicationType();
加载所有的初始化容器,设置应用上下文初始化器,从"META-INF/spring.factories"读取ApplicationContextInitializer类的实例名称集合并去重,并进行set去重。(一共5个)
this.setInitializers(this.getSpringFactoriesInstances(ApplicationContextInitializer.class));
设置监听器,从"META-INF/spring.factories"读取ApplicationListener类的实例名称集合并去重,并进行set去重。(一共10个)
this.setListeners(this.getSpringFactoriesInstances(ApplicationListener.class));
推断主入口应用类,通过当前调用栈,获取Main方法所在类,并赋值给mainApplicationClass
this.mainApplicationClass=this.deduceMainApplicationClass();
}
启动流程
springboot启动的运行方法,可以看到主要是各种运行环境的准备工作
public ConfigurableApplicationContext run(String...args){
创建并启动计时监控类
设置当前任务的id和启动的时间,方便后续的时候进行计时的操作
StopWatch stopWatch=new StopWatch();
stopWatch.start();
初始化上下文
初始化异常报告集合
ConfigurableApplicationContext context=null;
Collection<SpringBootExceptionReporter> exceptionReporters=new ArrayList();
设置系统属性“java.awt.headless”
默认为true,用于运行headless服务器,进行简单的图像处理,多用于在缺少显示屏、键盘或者鼠标时的系统配置,很多监控工具如jconsole 需要将该值设置为true
this.configureHeadlessProperty();
注册linsteners,创建所有spring运行监听器并发布应用启动事件,简单说的话就是获取SpringApplicationRunListener类型的实例(EventPublishingRunListener对象),并封装进SpringApplicationRunListeners对象,然后返回这个SpringApplicationRunListeners对象。说的再简单点,getRunListeners就是准备好了运行时监听器EventPublishingRunListener。
记载spring.factoris文件中的你内容
获取文件中对应类的全路径
根据反射得到具体的实体类对象
生成对应的对象之后再返回给调用者
SpringApplicationRunListeners listeners=this.getRunListeners(args);
每次监听器在实际做操作的时候都会执行listeners.starting()
this.initialMulticaster.multicastEvent
匹配不同类型的事件,然后从所有的监听器中把不符合条件的监听给过滤掉
符合条件的监听器会循环执行具体的自己的处理逻辑
listeners.starting(); try{
初始化默认参数应用类
加载命令行的参数值,解析在命令行中通过--key=value输入的属性值,封装到ApplicationArguments对象中
ApplicationArguments applicationArguments=new DefaultApplicationArguments(args);
根据运行监听器和应用参数来准备spring环境
ConfigurableEnvironment environment=this.prepareEnvironment(listeners,applicationArguments);
设置系统属性,保证某些bean不会添加到准备的环境中(设置忽略的参数)
this.configureIgnoreBeanInfo(environment);
创建banner打印类
Banner printedBanner=this.printBanner(environment);
创建应用上下文对象,根据当前应用类型选择创建什么类型的上下文 servlet,reactive,none
context=this.createApplicationContext();
设置异常报告对象,用来支持报告关于启动的错误
this.getSpringFactoriesInstances(SpringBootExceptionReporter.class,newClass[]{ConfigurableApplicationContext.class},context);
准备当前上下文对象,包含一个关键操作,将启动类注入容器,为后续开启自动化提供基础
设置初始化进行执行,向beanfactory中注入了三个postprocessor的对象,后续在自动装配的时候会用到applyInitializers(context);
listeners.contextPrepared(context);
加载很多资源配置,自动装配在此环节完成load(context, sources.toArray(new Object[0]));
listeners.contextLoaded(context);
this.prepareContext(context,environment,listeners,applicationArguments,printedBanner);
刷新应用上下文
this.refreshContext(context);
应用上下文刷新后置操作做一些扩展功能
this.afterRefresh(context,applicationArguments);
计时结束,并打印程序所用时长
stopWatch.stop();
输出日志,记录执行的主类目和事件信息
if(this.logStartupInfo){
(newStartupInfoLogger(this.mainApplicationClass)).logStarted(this.getApplicationLog(),stopWatch);
}
启动所有的监听器对象
listeners.started(context);
执行所有的runner运行器
this.callRunners(context,applicationArguments);
}catch(Throwablevar9){
处理失败的操作
this.handleRunFailure(context,listeners,exceptionReporters,var9);
thrownewIllegalStateException(var9);
}
启动完成调用,发布应用上下文就绪事件
listeners.running(context);
返回应用上下文
return context;
}
springboot执行流程的更多相关文章
- SpringBoot项目构建、测试、热部署、配置原理、执行流程
SpringBoot项目构建.测试.热部署.配置原理.执行流程 一.项目构建 二.测试和热部署 三.配置原理 四.执行流程
- SpringBoot启动流程解析
写在前面: 由于该系统是底层系统,以微服务形式对外暴露dubbo服务,所以本流程中SpringBoot不基于jetty或者tomcat等容器启动方式发布服务,而是以执行程序方式启动来发布(参考下图ke ...
- Spring Boot程序的执行流程
Spring Boot的执行流程如下图所示:(图片来源于网络) 上图为SpringBoot启动结构图,我们发现启动流程主要分为三个部分,第一部分进行SpringApplication的初始化模块,配置 ...
- Spring Security Oauth2 单点登录案例实现和执行流程剖析
Spring Security Oauth2 OAuth是一个关于授权的开放网络标准,在全世界得到的广泛的应用,目前是2.0的版本.OAuth2在“客户端”与“服务提供商”之间,设置了一个授权层(au ...
- Spring Security 案例实现和执行流程剖析
Spring Security Spring Security 是 Spring 社区的一个顶级项目,也是 Spring Boot 官方推荐使用的安全框架.除了常规的认证(Authentication ...
- SpringBoot启动流程分析(六):IoC容器依赖注入
SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...
- SpringBoot启动流程分析(二):SpringApplication的run方法
SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...
- SpringBoot启动流程分析(三):SpringApplication的run方法之prepareContext()方法
SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...
- SpringBoot启动流程分析(四):IoC容器的初始化过程
SpringBoot系列文章简介 SpringBoot源码阅读辅助篇: Spring IoC容器与应用上下文的设计与实现 SpringBoot启动流程源码分析: SpringBoot启动流程分析(一) ...
随机推荐
- Java多线程与线程池技术
一.序言 Java多线程编程线程池被广泛使用,甚至成为了标配. 线程池本质是池化技术的应用,和连接池类似,创建连接与关闭连接属于耗时操作,创建线程与销毁线程也属于重操作,为了提高效率,先提前创建好一批 ...
- DTO数据传输对象详解
文章目录 一.DTO是什么? 二.DTO解决的问题 三.代码演示 一.DTO是什么? DTO (数据传输对象) 数据传输对象(DTO),是一种设计模式之间传输数据的软件应用系统.数据传输目标往往是数据 ...
- 还原lvm逻辑卷创建整个过程
很多情况入职的时候,系统可能已规划过的,但是有的信息也不是很完整,比如下面的lvm逻辑卷我们先不管对与错,利用一些工具来了解当前lvm逻辑卷的情况 系统采样: [root@fp-web-112 var ...
- pt-osc又又出现死锁了
今天使用pt-osc修改mysql表结构,又出现死锁了,老大让尽量解决这个问题,我们先分析一下pt-osc容易出现死锁的原因,再来解决这个问题. 根据pt-osc打印的日志,可以看到pt-osc执行原 ...
- 从源码学习UEToll
从源码学习UEToll 捕获控件梳理 相对位置功能梳理 网格栅栏功能梳理 捕获代码分析 TransparentActivity public @interface Type { int TYPE_UN ...
- (转)Angular中的拦截器Interceptor
什么是拦截器? 异步操作 例子 Session 注入(请求拦截器) 时间戳(请求和响应拦截器) 请求恢复 (请求异常拦截) Session 恢复 (响应异常拦截器) 转之:http://my.osch ...
- FileNotFoundError: [Errno 2] No such file or directory: 'image/1.jpg'问题解决
FileNotFoundError: [Errno 2] No such file or directory: 'image/1.jpg'问题 最近在学习爬虫,想爬一些图片并保存到本地,但是在下载图片 ...
- 创建第一个c程序
创建,组织,生成 ,生成. 1.我们先创建一个win32项目. 文件->新建->项目->Visual C++ ->Win32 输入项目名称 选择项目保存位置 很重要的一 ...
- python和pycharm下载与安装
python解释器 1.python的由来 Python诞生于1989年的一个圣诞节,其创作者Guido van Rossum为了打发圣诞节假期的无聊,便开始了Python语言的编写.Python第一 ...
- Redis设计与实现3.3:集群
集群 这是<Redis设计与实现>系列的文章,系列导航:Redis设计与实现笔记 集群中的节点 创建集群 通过 CLUSTER NODE 命令可以查看当前集群中的节点.刚启动时,默认每一台 ...
