当前位置: 首页 > news >正文

从根儿上学习spring 四 之run方法启动第一段

图1

由上图我们可以看到,我把run方法分成了5个小段,每小段使用红框圈了起来,这一篇我们先开始讲第一段。大家需要关注下行号,我讲的时候可能会使用行号对应具体某行代码。

图1-289-290行:

没啥好说的定义了两个变量,191行的configureHeadlessProperty()方法仅仅是设置了系统属性java.awt.headless为true,告诉jvm使用无图形模式运行以节省资源,对我们研究spring没啥影响大家了解下即可。

图1-292行:

调用了getRunListeners(args); 方法返回了SpringApplicationRunListeners 对象,我们先看下SpringApplicationRunListeners对象是个啥,再看getRunListeners(args);方法干了啥事。

图2

上图便是SpringApplicationRunListeners类的部分方法和属性,由篇幅限制只能截取部分方法不过已经不影响我们对它的理解,有条件的再电脑旁的同学可以自己打开这个类看看。从上图很容易看出SpringApplicationRunListeners类主要维护了SpringApplicationRunListener的集合,并定义了一些监听方法,在方法内部调用SpringApplicationRunListener类的相同方法名的方法,从这些方法名大家应该也可以猜出这些方法大概是是什么时候调用的。比如starting应该是容器正在启动时调用的,environmentPrepared(ConfigurableEnvironment environment)方法是spring 环境environment准备好时调用的。

SpringApplicationRunListener监听器是处理springboot启动过程中的不同阶段的事件处理器大家需要和我们平时业务里经常使用的ApplicationListener监听器区分开。

我们看下getRunListeners(args)方法到底加载了哪些SpringApplicationRunListener。

图3

由上图我们可以看出逻辑在构建了types变量后主要逻辑在getSpringFactoriesInstances方法,我们看下getSpringFactoriesInstances方法

图4

由上图可以看到在getSpringFactoriesInstances方法里先获取了类加载器,然后使用SpringFactoriesLoader.loadFactoryNames(type, classLoader)方法是获取所有META-INF/spring.factories资源文件里的数据,META-INF/spring.factories文件里的内容大概是xxx=yyyy的形式,而这里的xxx就是type.getname()的值,yyyy就是获取到的names集合里的每个string值。而获取到的每个yyyy的值其实就是SpringApplicationRunListener接口的每个实现类的全限定名。具体的加载流程也不复杂大家可以自行看下,简单说就是把META-INF/spring.factories文件内容全读到map里然后使用key来获取对应的value,而value存的都是具体的实现类的类路径全限定名。

在获取所有的SpringApplicationRunListener接口的实现类string集合之后接下来就是实例化对象了,通过createSpringFactoriesInstances方法来实例化对象

图5

由上图5可知,在createSpringFactoriesInstances方法里遍历每个name,首先对每个name创建Class对象,再通过Class对象获取构造方法创建对象。由433--434行可知这里默认使用了有参构造器且参数类型是传过来的parameterTypes,由上文可知parameterTypes里分别是SpringApplication.class和string[].class,也就间接的说明spring要求SpringApplicationRunListener接口的实现类必须定义有参构造器并且第一个参数是SpringApplication,第二个参数是string数组。

图1-293行:

该行执行了SpringApplicationRunListeners的starting()方法,从上面我们对SpringApplicationRunListeners的剖析已经知道其实是调用了 SpringApplicationRunListener接口的starting()方法,那我们不难会有疑问这时候会有哪些默认的SpringApplicationRunListener的实现类呢?在293行打个断点我们可以看到这时候只有一个EventPublishingRunListener实现类,我们来看下该类的starting方法干了啥。

图6

由上图可知,在starting方法里调用了SimpleApplicationEventMulticaster对象的multicastEvent方法发布了ApplicationStartingEvent事件。我们知道在spring里监听事件一般都是ApplicationListener接口干的事,所以其实EventPublishingRunListener的starting方法就是发布了一个ApplicationStartingEvent事件由监听该事件的ApplicationListener去执行对应的onApplicationEvent方法。

这里我简单说下spring的监听器实现机制,主要包含三个模块分别是:1.事件发布器,2.事件监听器 3事件本身

也很好理解,在需要发布事件的地方先创建好事件,然后由事件发布器把事件发布出去,再由事件监听器来消费这些事件。而所谓的发布事件其实就是由事件发布器SimpleApplicationEventMulticaster维护了所有的监听器ApplicationListener,然后调用所有监听器ApplicationListener的onApplicationEvent方法罢了。

到这里我们稍微对第一段做个总结:

主要就是从META-INF/spring.factories文件里获取了SpringApplicationRunListener的实现类的定义并实例化,接着执行了其starting方法。而启动期间spring默认提供了EventPublishingRunListener实现类,在该类的stating方法里由调用了initialMulticaster.multicastEvent方法发布了ApplicationStartingEvent事件,由对应的ApplicationListener执行监听方法。

给大家留个小问题吧?在执行initialMulticaster.multicastEvent方法的时候ApplicationListener又是从哪获取的?

http://www.lryc.cn/news/415049.html

相关文章:

  • 智能闹钟如何判断用户已经醒了?
  • 【算法】动态规划解决背包问题
  • day09 工作日报表
  • C++学习之路(1)— 第一个HelloWorld程序
  • python3 pyside6图形库学习笔记及实践(三)
  • 03 库的操作
  • 嵌入式人工智能(44-基于树莓派4B的扩展板-LED按键数码管TM1638)
  • linux通过抓包工具tcpdump查看80端口访问量情况
  • Mac 上安装和卸载 SDKMAN 及管理多个 JDK
  • 字节测开一面面经
  • HTML 段落
  • 【Mind+】掌控板入门教程04 迷你动画片
  • 文件上传漏洞-HackBar使用
  • 鸿蒙媒体开发【相机数据采集保存】音频和视频
  • 【java基础】徒手写Hello, World!程序
  • 对 vllm 与 ollama 的一些研究
  • 浅谈基础的图算法——强联通分量算法(c++)
  • C#:通用方法总结—第13集
  • AI答题应用平台相关面试题
  • 树莓派NAS系统搭建教程:使用Flask和SQLite实现HTTP/HTTPS文件管理(代码示例)
  • mysql如何储存大量数据,分库存分表的建议和看法
  • Golang | Leetcode Golang题解之第310题最小高度树
  • 【面试系列】软件架构师 高频面试题及详细解答
  • 二百五十四、OceanBase——Linux上安装OceanBase数据库(四):登录ocp-express,配置租户管理等信息
  • HCIP学习作业一 | HCIA复习
  • OCR图片矫正、表格检测及裁剪综合实践
  • c++ 容器 vector
  • 零基础部署Minecraft到云服务器上教程
  • 常见cms漏洞之dedecms
  • 深入探究Liunx服务器内存:模拟程序实际占用与缓存占用内存