一、简介
1.前置知识
● Java17 ● Spring、SpringMVC、MyBatis ● Maven、IDEA
2.环境要求
环境&工具 版本(or later)
SpringBoot 3.1.x
IDEA 2023.x
Java 17+
Maven 3.5+
Tomcat 10.0+
Servlet 5.0+
GraalVM Community 22.3+
Native Build Tools 0.9.19+
二、SpringBoot3-核心原理
1.事件和监听器
1.1. 生命周期监听
1.2. 监听器-SpringApplicationRunListener
- 自定义SpringApplicationRunListener来监听事件;
- 编写SpringApplicationRunListener 实现类
**在 META-INF/spring.factories 中配置 org.springframework.boot.SpringApplicationRunListener=自己的Listener,还可以指定一个 有参构造器 ,接受两个参数(SpringApplication application, String[] args)
springboot 在spring-boot.jar中配置了默认的 Listener,如下
2. 生命周期
/** * Listener先要从 META-INF/spring.factories 读到 * * 1、引导: 利用 BootstrapContext 引导整个项目启动 * starting: 应用开始,SpringApplication的run方法一调用,只要有了 BootstrapContext 就执行 * environmentPrepared: 环境准备好(把启动参数等绑定到环境变量中),但是ioc还没有创建;【调一次】 * 2、启动: * contextPrepared: ioc容器创建并准备好,但是sources(主配置类)没加载。并关闭引导上下文;组件都没创建 【调一次】 * contextLoaded: ioc容器加载。主配置类加载进去了。但是ioc容器还没刷新(我们的bean没创建)。 * =======截止以前,ioc容器里面还没造bean呢======= * started: ioc容器刷新了(所有bean造好了),但是 runner 没调用。 * ready: ioc容器刷新了(所有bean造好了),所有 runner 调用完了。 * 3、运行 * 以前步骤都正确执行,代表容器running。 */
3.事件触发时机
1. 各种回调监听器
- BootstrapRegistryInitializer: 感知特定阶段:感知引导初始化
META-INF/spring.factories
创建引导上下文bootstrapContext的时候触发。
application.addBootstrapRegistryInitializer();
场景:进行密钥校对授权。
- ApplicationContextInitializer: 感知特定阶段: 感知ioc容器初始化
META-INF/spring.factories
application.addInitializers();
- ApplicationListener: 感知全阶段:基于事件机制,感知事件。 一旦到了哪个阶段可以做别的事
@Bean或@EventListener: 事件驱动
SpringApplication.addListeners(…)或 SpringApplicationBuilder.listeners(…)
META-INF/spring.factories
- SpringApplicationRunListener: 感知全阶段生命周期 + 各种阶段都能自定义操作; 功能更完善。
META-INF/spring.factories
- ApplicationRunner: 感知特定阶段:感知应用就绪Ready。卡死应用,就不会就绪
@Bean
- CommandLineRunner: 感知特定阶段:感知应用就绪Ready。卡死应用,就不会就绪
@Bean
最佳实战:
- 如果项目启动前做事: BootstrapRegistryInitializer 和 ApplicationContextInitializer
- 如果想要在项目启动完成后做事:**ApplicationRunner**和 **CommandLineRunner**
- 如果要干涉生命周期做事:**SpringApplicationRunListener**
- 如果想要用事件机制:**ApplicationListener**
2. 完整触发流程
9大事件触发顺序&时机
- ApplicationStartingEvent:应用启动但未做任何事情, 除过注册listeners and initializers.
- ApplicationEnvironmentPreparedEvent: Environment 准备好,但context 未创建.
- ApplicationContextInitializedEvent: ApplicationContext 准备好,ApplicationContextInitializers 调用,但是任何bean未加载
- ApplicationPreparedEvent: 容器刷新之前,bean定义信息加载
- ApplicationStartedEvent: 容器刷新完成, runner未调用
=以下就开始插入了探针机制====
- AvailabilityChangeEvent: LivenessState.CORRECT应用存活; 存活探针
- ApplicationReadyEvent: 任何runner被调用
- AvailabilityChangeEvent:ReadinessState.ACCEPTING_TRAFFIC就绪探针,可以接请求
- ApplicationFailedEvent:启动出错
应用事件发送顺序如下:
感知应用是否存活了:可能植物状态,虽然活着但是不能处理请求。
应用是否就绪了:能响应请求,说明确实活的比较好。
3. SpringBoot 事件驱动开发
应用启动过程生命周期事件感知(9大事件)、应用运行中事件感知(无数种)。
- 事件发布:ApplicationEventPublisherAware或注入:ApplicationEventMulticaster
- 事件监听:组件 + @EventListener
事件发布者
@Service public class EventPublisher implements ApplicationEventPublisherAware { /** * 底层发送事件用的组件,SpringBoot会通过ApplicationEventPublisherAware接口自动注入给我们 * 事件是广播出去的。所有监听这个事件的监听器都可以收到 */ ApplicationEventPublisher applicationEventPublisher; /** * 所有事件都可以发 * @param event */ public void sendEvent(ApplicationEvent event) { //调用底层API发送事件 applicationEventPublisher.publishEvent(event); } /** * 会被自动调用,把真正发事件的底层组组件给我们注入进来 * @param applicationEventPublisher event publisher to be used by this object */ @Override public void setApplicationEventPublisher(ApplicationEventPublisher applicationEventPublisher) { this.applicationEventPublisher = applicationEventPublisher; } }
事件订阅者
@Service public class CouponService { @Order(1) @EventListener public void onEvent(LoginSuccessEvent loginSuccessEvent){ System.out.println("===== CouponService ====感知到事件"+loginSuccessEvent); UserEntity source = (UserEntity) loginSuccessEvent.getSource(); sendCoupon(source.getUsername()); } public void sendCoupon(String username){ System.out.println(username + " 随机得到了一张优惠券"); } }