接上文 Spring Developer Tools 源码分析:二、类路径监控,接下来看看前面提到的这些类是如何配置,如何启动的。

spring-boot-devtools 使用了 Spring Boot 的自动配置方式,我们先关注本地开发环境中自动重启的部分。

在 LocalDevToolsAutoConfiguration 主要包含了 LiveReload 和重启的配置,LiveReload 后续看情况再介绍,这里先看重启的配置。

3.1 LocalDevToolsAutoConfiguration 本地配置

在这个类上 @ConditionalOnInitializedRestarter 是一个该配置生效的条件,具体实现中会判断 Restarter 是否已经实例化,并且是否存在可被监控的类目录(除 jar 文件外的目录),如果已经实例化,并且有需要监控的目录才会启动。

通过 java -jar 方式启动的时候,由于不存在需要监控的目录,devtools 不会触发后续的配置,因此虽然启动了 Restarter,但是并不会监控目录,也不会自动重启。

当通过 IDE 启动项目时,默认情况下会满足这里的条件,LocalDevToolsAutoConfiguration 中的其他配置可以生效。

3.2 RestartConfiguration 重启配置

在 RestartConfiguration 上也有限制条件,只有当设置下面的参数时才不会生效


不设置或者设置为任何不是 false (忽略大小写)的值时,都会生效。

除了这个参数能控制外,还有一个可以直接控制 Restarter 是否生效的参数,后续会介绍。

下面逐个看 RestartConfiguration 中的各个配置。

3.2.1 FileSystemWatcherFactory 文件监控

配置代码如下:


FileSystemWatcherFactory 是一个函数式接口,这里直接返回了一个方法引用。将来调用该接口的方法时,就会执行下面的 newFileSystemWatcher 方法。

每次调用 fileSystemWatcherFactory() 方法时,返回的都是同一个 FileSystemWatcherFactory,但是调用工厂的 getFileSystemWatcher() 方法时返回都是新的 FileSystemWatcher。

通过这个方法可以看到创建 FileSystemWatcher 时,这里会判断是否配置了触发文件(只有修改指定文件才会重启),是否配置额外需要监控变化的位置。Java架构交流学习圈:874811168 面向1-3年经验 Java开发人员 帮助突破瓶颈 提升思维能力

3.2.2 ClassPathRestartStrategy 重启策略

代码如下:


该方法返回了 PatternClassPathRestartStrategy 实现类,判断是否重启时,忽略所有传入的位置,默认忽略的位置如下:


额外的排除项可以通过下面的参数设置:


3.2.3 ClassPathFileSystemWatcher 类路径监控

代码如下:


这里创建时,从 Restarter 获取了需要监控的类路径(后续会详细介绍),然后创建了一个 watcher,使用了前面创建的两个 bean。setStopWatcherOnRestart 的意思是当类路径发生变化并且需要重启时,是否停止类监控,这里设置了 true,也就是重启前会停止监控。

如果重启前会停止监控,我们可能需要担心如果配置改错了导致 Spring 无法启动该怎么办,devtools 也提供了相应的策略来解决这个问题,下一小节就会看到。

3.2.3 监听 ClassPathChangedEvent

代码如下:


当一个方法添加 @EventListener 时,Spring 会把该方法添加到事件监听中,当触发该事件时,这个方法就会被调用。在上一篇介绍了 ClassPathChangedEvent,其中包含了 restartRequired,这里会判断是否需要重启,当需要重启时,就会调用 Restarter 的实例进行重启。

在重启方法中还传入了 FileWatchingFailureHandler 并且使用了 fileSystemWatcherFactory。

FileWatchingFailureHandler 实现了 FailureHandler 接口,该接口用于在重启的启动过程中,如果出错了,要采取什么策略去进行下去,该接口方法返回的结果只有两种,Outcome.ABORT 中止或 Outcome.RETRY 重试。

FileWatchingFailureHandler 中会通过 fileSystemWatcherFactory 创建一个新的文件监控,当类路径的内容发生变化时(不需要考虑是否需要重启,因为已经停止,并且没启动成功)就尝试重新启动。这种策略可以解决当配置或者代码出错无法启动时,可以修改BUG解决错误,然后 devtools 自动尝试启动。

FileWatchingFailureHandler 中的 FileSystemWatcher 和 3.2.3 中的不是同一个,并且在重启前另一个已经关闭,关闭的目的不是为了防止和这里存在两个监控出现冲突,而是为了防止第一次修改后,还没有重启时又发生了变化,由于重启需要时间,这就会导致重启还没完成就又重启了,这种情况下除了会产生错误外,还会因为短时间内频繁重启导致重启时间过长。Java架构交流学习圈:874811168 面向1-3年经验 Java开发人员 帮助突破瓶颈 提升思维能力

在重启时,会先关闭所有的 Spring Context,此时也会触发 ClassPathFileSystemWatcher 中的 destroy 方法:


在 destroy 中也会关闭文件监控,所以无论如何都不会和这里的 FileSystemWatcher 产生冲突。

3.3 重启后会重新初始化 Spring

关闭再启动时,devtools 通过反射执行的我们自己的 XXApplication 类的 main 方法,因此 LocalDevToolsAutoConfiguration 也会重新初始化,类路径监控也会重新建立。我们现在只是了解了从文件监控、类路径监控以及监控配置启动和触发重启的过程。有很多关键的处理过程和 RestartClassLoader 以及 Restarter 有关,为了防止过长篇幅使得关注点太过分散,后续会分别介绍这两部分内容。

转载于:https://blog.51cto.com/13982920/2312175

Spring Developer Tools 源码分析:三、重启自动配置'的更多相关文章

  1. Spring Developer Tools 源码分析:二、类路径监控

    在 Spring Developer Tools 源码分析一中介绍了 devtools 提供的文件监控实现,在第二部分中,我们将会使用第一部分提供的目录监控功能,实现对开发环境中 classpath ...

  2. Spring IOC 容器源码分析 - 填充属性到 bean 原始对象

    1. 简介 本篇文章,我们来一起了解一下 Spring 是如何将配置文件中的属性值填充到 bean 对象中的.我在前面几篇文章中介绍过 Spring 创建 bean 的流程,即 Spring 先通过反 ...

  3. Spring IOC 容器源码分析 - 循环依赖的解决办法

    1. 简介 本文,我们来看一下 Spring 是如何解决循环依赖问题的.在本篇文章中,我会首先向大家介绍一下什么是循环依赖.然后,进入源码分析阶段.为了更好的说明 Spring 解决循环依赖的办法,我 ...

  4. Spring IOC 容器源码分析 - 创建原始 bean 对象

    1. 简介 本篇文章是上一篇文章(创建单例 bean 的过程)的延续.在上一篇文章中,我们从战略层面上领略了doCreateBean方法的全过程.本篇文章,我们就从战术的层面上,详细分析doCreat ...

  5. Spring IOC 容器源码分析 - 创建单例 bean 的过程

    1. 简介 在上一篇文章中,我比较详细的分析了获取 bean 的方法,也就是getBean(String)的实现逻辑.对于已实例化好的单例 bean,getBean(String) 方法并不会再一次去 ...

  6. 十、Spring之BeanFactory源码分析(二)

    Spring之BeanFactory源码分析(二) 前言 在前面我们简单的分析了BeanFactory的结构,ListableBeanFactory,HierarchicalBeanFactory,A ...

  7. 九、Spring之BeanFactory源码分析(一)

    Spring之BeanFactory源码分析(一) ​ 注意:该随笔内容完全引自https://blog.csdn.net/u014634338/article/details/82865644,写的 ...

  8. tomcat源码分析(三)一次http请求的旅行-从Socket说起

    p { margin-bottom: 0.25cm; line-height: 120% } tomcat源码分析(三)一次http请求的旅行 在http请求旅行之前,我们先来准备下我们所需要的工具. ...

  9. Spring IOC 容器源码分析 - 余下的初始化工作

    1. 简介 本篇文章是"Spring IOC 容器源码分析"系列文章的最后一篇文章,本篇文章所分析的对象是 initializeBean 方法,该方法用于对已完成属性填充的 bea ...

随机推荐

  1. Redis 笔记(一)——数据类型简介

    Redis 是一个 key-value 存储系统,但是它的 value 值不仅仅可以存储字符串,value 共有 五种 数据结构类型,具体如下: 数据结构类型 结构类型 结构存储的值 结构的读写能力 ...

  2. 剑指offer—单链表反转的三种实现方法

    单链表的反转可以用递归.非递归和栈的方法实现 链表节点定义: struct ListNode{ int val; Node* next; ListNode(int x):val(x),next(nul ...

  3. 字符串操作 — Google Guava

    前言 Java 里字符串表示字符的不可变序列,创建后就不能更改.在我们日常的工作中,字符串的使用非常频繁,熟练的对其操作可以极大的提升我们的工作效率,今天要介绍的主角是 Google 开源的一个核心 ...

  4. scratch 如何改变变量的作用域

    在新建变量的时候,有个选项是“适用于所有角色”还是“仅适用于当前角色”.通常称前者为全局变量,所有角色都可以访问到这个变量:后者,称为局部变量,只能在当前角色里访问到这个变量.例如,在使用克隆功能时, ...

  5. python通俗讲解闭包

    通俗理解闭包 先来看看什么是闭包吧 闭包是引用了自由变量的函数.这个被引用的自由变量将和这个函数一同存在,即使已经离开了创造它的环境也不例外.所以,有另一种说法认为闭包是由函数和与其相关的引用环境组合 ...

  6. Linux管理,例行性工作,at和cron

    at 仅执行一次就结束调度(服务atd必须启动) at的配置信息 1.我们设定的at进程其实都是被以文本的方式写入到/var/spool/at/目录内,然后这些设定的进程便会等待atd服务来启动这些进 ...

  7. pyecharts的使用及总结

    包的下载及配置 这个包的相应的配置较多,版本也不兼容,总结一下 预览:pyecharts画图 pip pyecharts pip 各级别地图(6.7个左右) pip jupyter环境 [为了生成pn ...

  8. TP5快速入门

    一.查询 //order支持使用数组对多个字段的排序,例如order(['order','id'=>'desc']) //group方法只有一个参数,并且只能使用字符串. //having方法只 ...

  9. sigmod函数求导

    sigmod函数: \[f(z)=\frac{1}{1+e^{-z}} \] 求导: \[\frac{\partial f(z)}{\partial z}=\frac{-1*-1*e^{-z}}{(1 ...

  10. hibernate.current_session_context_class 比较权威的解释

    hibernate.current_session_context_class 博客分类: hibernate HibernateSpring多线程配置管理thread  遇到过的问题: 情景1: 在 ...