SpringBoot ApplicationEvent详解
ApplicationStartingEvent 阶段
LoggingApplicationListener#onApplicationStartingEvent
初始化日志工厂,LoggingSystemFactory接口,可以通过spring.factories进行定制
可以通过System.setProperty("org.springframework.boot.logging.LoggingSystem","类全路径限定名或者none") 指定log实现类
如果配置的是none,则返回 org.springframework.boot.logging.LoggingSystem.NoOpLoggingSystem
springboot默认指定了三种类型:
org.springframework.boot.logging.logback.LogbackLoggingSystem.Factory
org.springframework.boot.logging.log4j2.Log4J2LoggingSystem.Factory
org.springframework.boot.logging.java.JavaLoggingSystem.Factory
----------
BackgroundPreinitializer#onApplicationEvent 在当前阶段不做任何处理!!!
可以通过System.setProperty("spring.backgroundpreinitializer.ignore","true|false")
同时满足服务器是多核cpu并且非GraalVM环境
来指定是否通过后台线程去加载某些资源,默认是单独开一个线程来加载某些资源
线程名称:background-preinit
后台加载的资源:
ConversionServiceInitializer.class
ValidationInitializer.class
MessageConverterInitializer.class
JacksonInitializer.class
CharsetInitializer.class
有异常直接忽略
----------
DelegatingApplicationListener#onApplicationEvent 在当前阶段不做任何处理!!!
可以通过配置 context.listener.classes 属性来指定要执行的listener,是一个复合包装类
内部定义了SimpleApplicationEventMulticaster事件驱动类,用来指定配置的listener
==========
ApplicationEnvironmentPreparedEvent 阶段
EnvironmentPostProcessorApplicationListener#onApplicationEvent
onApplicationEnvironmentPreparedEvent((ApplicationEnvironmentPreparedEvent) event);
获取ConfigurableEnvironment实例
获取SpringApplication实例
通过SpringApplication实例.getResourceLoader()和ConfigurableEnvironment实例.getBootstrapContext() 获取 EnvironmentPostProcessors
所有实现了 org.springframework.boot.env.EnvironmentPostProcessor 接口的实现类,同样可以通过 spring.factories进行定制
开始遍历
1、RandomValuePropertySourceEnvironmentPostProcessor#postProcessEnvironment
将RandomValuePropertySource添加到systemEnvironment这个属性解析器集合的最后一位
RandomValuePropertySource: 用法,配置属性以random开头
比如:random.int 返回正负32位以内的一个伪随机数字
random.long 返回正负64位以内的一个伪随机数字
random.int(0,9) 包含0不包含9 中的一个伪随机数字
random.long[0,9] 包含0不包含9 中的一个伪随机数字
() 和 [] 或者 $$ 作用都一样 前后任意两个字符括起来,数字使用逗号分隔即可
2、SystemEnvironmentPropertySourceEnvironmentPostProcessor#postProcessEnvironment
判断是否设置了 SpringApplication.environmentPrefix 属性,如果设置了将systemEnvironment(SystemEnvironmentPropertySource)
的数据封装为OriginAwareSystemEnvironmentPropertySource类(SystemEnvironmentPropertySource的子类),有prefix属性.
spring.main.environment-prefix:不允许这样设置
可以通过 SpringApplicationBuilder.environmentPrefix("xx") 来设置,可以以 . - _ 等结尾,
3、SpringApplicationJsonEnvironmentPostProcessor#postProcessEnvironment
解析json属性,使用spring.application.json或者SPRING_APPLICATION_JSON命令行参数指定,按序返回有效的第一个进行解析.
解析成功后,将json属性封装为JsonPropertySource,放到servlet sources前面,如果不是servlet环境,则放在第一位
解析json属性的解析器:
org.springframework.boot.json.JsonParserFactory#getJsonParser: 指定了4种json解析器
com.fasterxml.jackson.databind.ObjectMapper
com.google.gson.Gson
org.yaml.snakeyaml.Yaml
org.springframework.boot.json.BasicJsonParser
按以上顺序进行加载,加载成功就返回对应的对象实例
4、CloudFoundryVcapEnvironmentPostProcessor#postProcessEnvironment 在springcloud环境下生效
判断是否设置了 spring.main.cloud-platform VCAP_APPLICATION VCAP_SERVICES 满足任意一个
设置了 添加 vcap sources
5、ConfigDataEnvironmentPostProcessor#postProcessEnvironment **你个** 加载并且解析设定的配置文件
spring.config.on-not-found: 配置找不到的处理方法,参考ConfigDataNotFoundAction枚举类
SpringApplication.additionalProfiles 通过 builder 构建
加载指定的配置文件并且设置environment中的profiles
# ConfigData Location Resolvers
org.springframework.boot.context.config.ConfigDataLocationResolver=\
org.springframework.boot.context.config.ConfigTreeConfigDataLocationResolver,\
org.springframework.boot.context.config.StandardConfigDataLocationResolver
# ConfigData Loaders
org.springframework.boot.context.config.ConfigDataLoader=\
org.springframework.boot.context.config.ConfigTreeConfigDataLoader,\
org.springframework.boot.context.config.StandardConfigDataLoader
6、DebugAgentEnvironmentPostProcessor#postProcessEnvironment
reactor.tools.agent.ReactorDebugAgent有这个类并且spring.reactor.debug-agent.enabled属性配置为true时
执行 ReactorDebugAgent init() 方法
7、IntegrationPropertiesEnvironmentPostProcessor#postProcessEnvironment
META-INF/spring.integration.properties 这个文件存在时加载内容并且转化为IntegrationPropertiesPropertySource添加到sources结尾
----------
AnsiOutputApplicationListener#onApplicationEvent
进行属性绑定 spring.output.ansi.enabled org.springframework.boot.ansi.AnsiOutput#enabled 参考 Enabled 枚举类
spring.output.ansi.console-available AnsiOutput.consoleAvailable = consoleAvailable
----------
LoggingApplicationListener#onApplicationEvent
日志文件和属性初始化配置
----------
BackgroundPreinitializer#onApplicationEvent
可以通过System.setProperty("spring.backgroundpreinitializer.ignore","true|false")
同时满足服务器是多核cpu并且非GraalVM环境
来指定是否通过后台线程去加载某些资源,默认是单独开一个线程来加载某些资源
线程名称:background-preinit
后台加载的资源:
ConversionServiceInitializer.class
ValidationInitializer.class
MessageConverterInitializer.class
JacksonInitializer.class
CharsetInitializer.class
有异常直接忽略
----------
DelegatingApplicationListener#onApplicationEvent
可以通过配置 context.listener.classes 属性来指定要执行的listener,是一个复合包装类
内部定义了SimpleApplicationEventMulticaster事件驱动类,用来指定配置的listener
----------
FileEncodingApplicationListener#onApplicationEvent1
spring.mandatory-file-encoding: 查看是否配置了此属性,强制编码,如果这个与file.encoding不符合报错!!!
==========
bindToSpringApplication(environment) // 将spring.main开头的属性配置绑定到SpringApplication属性上
==========
applyInitializers(context); // 执行ApplicationContextInitializer接口的实现类
==========
ApplicationContextInitializedEvent
BackgroundPreinitializer#onApplicationEvent 此阶段啥也不做!!!
----------
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
==========
ApplicationPreparedEvent
EnvironmentPostProcessorApplicationListener#onApplicationEvent
onApplicationPreparedEvent() > EnvironmentPostProcessorApplicationListener#finish() > DeferredLogs#switchOverAll()
打印日志
----------
LoggingApplicationListener#onApplicationEvent
注册springBootLoggingSystem单例Bean
logFile存在并且springBootLogFile不存在这个Bean时注册springBootLogFile单例Bean
loggerGroups存在并且springBootLoggerGroups不存在这个Bean时注册springBootLoggerGroups单例Bean
springBootLoggingLifecycle单例Bean不存在BeanFactory.getParent为空时注册springBootLoggingLifecycle单例Bean
----------
BackgroundPreinitializer#onApplicationEvent 此阶段啥也不做!!!
----------
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
==========
中间存在的各种事件驱动类
ServletWebServerInitializedEvent
SpringApplicationAdminMXBeanRegistrar#onApplicationEvent : onWebServerInitializedEvent((WebServerInitializedEvent) event);
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
ServerPortInfoApplicationContextInitializer#onApplicationEvent : 绑定server.ports sources
----------
ContextRefreshedEvent
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
ConditionEvaluationReportLoggingListener.ConditionEvaluationReportListener#onApplicationEvent 打印方法 ConditionEvaluationReportMessage
ClearCachesApplicationListener#onApplicationEvent 清理加载反射field和method的缓存数据,调用类加载器的clearCache方法清理缓存
SharedMetadataReaderFactoryContextInitializer.SharedMetadataReaderFactoryBean#onApplicationEvent 清理加载的class缓存数据
ResourceUrlProvider#onApplicationEvent 静态资源 /webjars/** 和 /static/**
==========
ApplicationStartedEvent
BackgroundPreinitializer#onApplicationEvent 此阶段啥也不做!!!
----------
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
----------
StartupTimeMetricsListener#onApplicationEvent
设置埋点监控 TimeGauge
----------
TomcatMetricsBinder#onApplicationEvent
设置tomcat监控绑定
----------
AvailabilityChangeEvent
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
----------
ApplicationAvailabilityBean#onApplicationEvent 添加到 org.springframework.boot.availability.ApplicationAvailabilityBean#events 集合中 应用启动好了
==========
ApplicationReadyEvent
SpringApplicationAdminMXBeanRegistrar#onApplicationEvent 应用准备好了
----------
BackgroundPreinitializer#onApplicationEvent 此阶段啥也不做!!!
----------
StartupTimeMetricsListener#onApplicationEvent 注册TimeGauge埋点监控
----------
DelegatingApplicationListener#onApplicationEvent 此阶段啥也不做!!!
==========
AvailabilityChangeEvent
ApplicationAvailabilityBean#onApplicationEvent 添加到 org.springframework.boot.availability.ApplicationAvailabilityBean#events 集合中 应用准备好了
相关文章:
SpringBoot ApplicationEvent详解
ApplicationStartingEvent 阶段 LoggingApplicationListener#onApplicationStartingEvent 初始化日志工厂,LoggingSystemFactory接口,可以通过spring.factories进行定制 可以通过System.setProperty("org.springframework.boot.logging.LoggingSystem",&q…...
WebSocket 报java.io.IOException: 远程主机强迫关闭了一个现有的连接。
在客户端强制关闭时,或者窗口强制关闭时,后端session没有关闭。 有时还会报:java.io.EOFException: 这个异常 前端心跳没有收到信息,还在心跳。 CloseReason close new CloseReason(CloseReason.CloseCodes.NORMAL_CLOSURE, &…...
关于git约定式提交IDEA
背景 因为git提交的消息不规范导致被乱喷,所以领导统一规定了约定式提交 官话 约定式提交官网地址 约定式提交规范是一种基于提交信息的轻量级约定。 它提供了一组简单规则来创建清晰的提交历史; 这更有利于编写自动化工具。 通过在提交信息中描述功能…...
【计算机网络】http协议
目录 前言 认识URL URLEncode和URLDecode http协议格式 http方法 GET POST GET与POST的区别 http状态码 http常见header 简易的http服务器 前言 我们在序列化和反序列化这一章中,实现了一个网络版的计算器。这个里面设计到了对协议的分析与处…...
仓库太大,clone 后,git pull 老分支成功,最新分支失败
由于 git 仓库太大,新加入的小伙伴在拉取时,无法切换到最新的分支,报错如下: fetch-pack: unexpected disconnect while reading sideband packet fatal: early EOF fatal: fetch-pack: invalid index-pack output在此记录解决步…...
javafx Dialog无法关闭
// 生成二维码图片String qrCodeText "https://example.com";DialogPane grid new DialogPane();grid.setPadding(new Insets(5));VBox vBox new VBox();vBox.setAlignment(Pos.CENTER);Image qrCodeImage generateQRCodeImage(qrCodeText);ImageView customImag…...
vue3中TCplayer应用
环境win10:vitevue3elementUI 1 安装 npm install tcplayer.js2 使用 <template><div><video id"player-container-id" width"414" height"270" preload"auto" playsinline webkit-playsinline></video>&l…...
算法通关村14关 | 数据流中位数问题
1. 数据流中位数问题 题目 LeetCode295: 中位数是有序列表中间的数,如果列表长度是偶数,中位数是中间两个数的平均值, 例如:[2,3,4]的中位数是3, [2,3]中位数是(23)/ 2 2.5 设计一个数据结构: …...
工厂模式 与 抽象工厂模式 的区别
工厂模式: // 抽象产品接口 interface Product {void showInfo(); }// 具体产品A class ConcreteProductA implements Product {Overridepublic void showInfo() {System.out.println("This is Product A");} }// 具体产品B class ConcreteProductB impl…...
安装虚拟机+安装/删除镜像
安装虚拟机 注意,官网可能无法登录,导致无法从官网下载,就自己去网上搜靠谱的下载,我用的16.2.3 删除镜像 Vm虚拟机怎么删除已经创建的系统?Vm虚拟机创建好之后iso删除方法 - 系统之家 (xitongzhijia.net) 安装镜像…...
MySQL的内置函数复合查询内外连接
文章目录 内置函数时间函数字符串函数数学函数其他函数 复合查询多表笛卡尔积自连接在where中使用子查询多列子查询在from中使用子查询 内连接外连接左外连接右外连接 内置函数 时间函数 函数描述current_date()当前日期current_time()当前时间current_timestamp()当前时间戳…...
操作系统(OS)与系统进程
操作系统(OS)与系统进程 冯诺依曼体系结构操作系统(Operator System)进程基本概念进程的描述(PCB)查看进程通过系统调用获取进程标示符(PID)通过系统调用创建进程(fork)进程状态&…...
防重复提交:自定义注解 + 拦截器(HandlerInterceptor)
防重复提交:自定义注解 拦截器(HandlerInterceptor) 一、思路: 1、首先自定义注解; 2、创建拦截器实现类(自定义类名称),拦截器(HandlerInterceptor); 3…...
Excel中将文本格式的数值转换为数字
在使用excel时,有时需要对数字列进行各种计算,比如求平均值,我们都知道应该使用AVERAGE()函数,但是很多时候结果却“不尽如人意”。 1 问题: 使用AVERAGE函数: 结果: 可以看到单元格左上角有个…...
uni-app开发小程序中遇到的map地图的点聚合以及polygon划分区域问题
写一篇文章来记录以下我在开发小程序地图过程中遇到的两个小坑吧,一个是点聚合,用的是joinCluster这个指令,另一个是polygon在地图上划分多边形的问题: 1.首先说一下点聚合问题,由于之前没有做过小程序地图问题&#…...
【笔记】软件测试的艺术
软件测试的心理学和经济学 测试是为发现错误而执行程序的过程,所以它是一个破坏性的过程,测试是一个“施虐”的过程。 软件测试的10大原则 1、测试用例需要对预期输出的结果有明确的定义 做这件事的前提是能够提前知晓需求和效果图,如果不…...
配置本地maven
安装maven安装包 修改环境变量 vim ~/.bash_profile export JMETER_HOME/Users/yyyyjinying/apache-jmeter-5.4.1 export GOROOT/usr/local/go export GOPATH/Users/yyyyjinying/demo-file/git/backend/go export GROOVY_HOME/Users/yyyyjinying/sortware/groovy-4.0.14 exp…...
C# 按钮的AcceptButton和CancelButton属性
using System; using System.Collections.Generic; using System.ComponentModel; using System.Data; using System.Drawing; using System...
SMT贴片制造:专业、现代、智能的未来之选
在现代科技的快速发展下,SMT贴片制造作为电子元器件的核心工艺之一,正以其专业、现代和智能的特点成为未来的首选。 随着电子产品越来越小型化,传统的手工焊接已经无法满足高速、高精度、高稳定性的要求。而SMT贴片制造作为一种先进的表面贴…...
python sqlalchemy db.session 的commit()和colse()对session中的对象的影响
实验一:commit()之后查看stu的属性id,查看db.session是否改变 db_test.route("/db_test",methods["GET"]) def db_test():stuStuTest()stu.stu_age22stu.stu_name"nnannns"stu.stu_class11print("sessio…...
007、大语言模型集成:Prompt工程与上下文管理
昨天深夜调试时遇到一个诡异问题:同样的查询,在本地测试时LLM能准确返回产品参数,上了生产环境就总答非所问。盯着监控日志看了半小时才发现,某个微服务在拼接用户历史对话时,漏掉了两条关键消息——上下文窗口看似饱满,实则缺了核心信息。这个坑让我重新审视了RAG系统中…...
WarcraftHelper:魔兽争霸III终极优化指南 - 解决宽屏、帧率、地图限制三大痛点
WarcraftHelper:魔兽争霸III终极优化指南 - 解决宽屏、帧率、地图限制三大痛点 【免费下载链接】WarcraftHelper Warcraft III Helper , support 1.20e, 1.24e, 1.26a, 1.27a, 1.27b 项目地址: https://gitcode.com/gh_mirrors/wa/WarcraftHelper 你是否还在…...
告别命令行手敲:用Python脚本自动化你的第一个OpenFOAM腔体流动模拟
用Python脚本解放双手:OpenFOAM腔体流动模拟自动化实战 每次打开终端,重复输入相同的OpenFOAM命令,修改几乎雷同的参数文件,这种机械操作是否让你感到效率低下?作为CFD工程师,我们真正应该投入时间的是分析…...
Claude Code 进行“从头重写”的项目 Claw Code全面介绍(claw-code)
一个对泄露的 Claude Code 进行“从头重写”的项目,已成为 GitHub 上增长最快的仓库。一、项目是什么 claw-code(仓库全名写作 Rewriting Project Claw Code)是一个围绕「智能体 Harness( harness:把模型、工具、会话与…...
别再用默认源了!Ubuntu22.04换源后软件下载速度提升10倍的秘密
别再用默认源了!Ubuntu22.04换源后软件下载速度提升10倍的秘密 当你在Ubuntu终端里输入apt update后盯着缓慢爬升的进度条发呆时,有没有想过这背后隐藏着一个影响开发效率的关键因素?作为长期使用Ubuntu的开发老鸟,我发现90%的用户…...
1111111111111111111111
111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111...
本地AI聊天、交互助手(写给小白的LLM工具选型系列:第三篇)
诸神缄默不语-个人技术博文与视频目录 在这一章介绍的是,已经有了AI大模型推理服务(不管是云端API还是本地服务),想要一个像聊天框那样的界面来跟大模型聊天、或者让大模型做更复杂的工作。 本章主要考虑的功能还是AI对话&#x…...
nwb最佳实践指南:10个提升React、Preact和Inferno开发效率的关键技巧
nwb最佳实践指南:10个提升React、Preact和Inferno开发效率的关键技巧 【免费下载链接】nwb A toolkit for React, Preact, Inferno & vanilla JS apps, React libraries and other npm modules for the web, with no configuration (until you need it) 项目地…...
MySQL函数及条件查询相关用法
文章目录 前言 一、函数(可跳过) 1.字符串函数 2.数值函数 3.日期和时间函数 4.聚合函数(常用) 5.控制流函数 6.加密和压缩函数 7.系统信息函数 二、条件查询(select) 1.筛选条件子句where与hav…...
别再怕环路!手把手教你用锐捷RG-IS2700G交换机配置ERPS环网(附完整命令)
锐捷RG-IS2700G交换机ERPS环网实战:从零搭建高可靠企业网络 第一次接手企业园区网核心交换机的运维工作时,看到拓扑图上那个醒目的环形结构,我的手指在键盘上方悬停了整整十分钟——毕竟谁都不想成为"那个让全公司断网的新人"。直到…...
