文章摘要(AI生成)
Spring 中的应用容器 ApplicationContext 在管理 Bean 的过程中提供了更多的功能和灵活性,与传统的 Bean 工厂相比,它实现了资源加载器、环境变量、国际化消息、事件发布器、生命周期等功能接口。通过创建应用容器,我们可以实现环境变量验证、SPEL 属性编辑器、容器级后置处理器、国际化消息、事件发布和监听、生命周期监控等功能。应用容器启动时会调用 refresh 方法,依次进行容器初始化、环境变量验证、Bean 工厂初始化、自定义容器操作等流程。在关闭时,需要发布关闭事件、销毁单例、关闭 Bean 工厂等操作。通过监听容器事件可以实现对容器状态的监听,也可以创建自定义容器并重写相应方法。通过应用容器的功能,我们可以实现更多定制化的操作和管理方式。
诸如前文bean获取&bean工厂职责梳理所述,spring通过bean工厂替我们管理了我们平时开发中对象创建和属性填充等工作,但是我们实际在开发中,并没有直接使用XmlBeanFactory
进行bean的获取或其流程的操作,而是通过创建应用容器(ClassPathXmlApplicationContext beanFactory = new ClassPathXmlApplicationContext("spring/application-context.xml");
)的方式获取bean。
那么这个ClassPathXmlApplicationContext
与bean工厂又什么不同,我们又可以通过它做到哪些bean工厂无法实现的功能呢?
初识应用容器ApplicationContext
首先我们看下应用容器的层级关系:
可以发现应用容器接口在实现了我们所熟知的bean工厂之外,还实现了以下功能接口:
- 资源加载器-获取相应的类加载器和资源文件
- 环境变量-拓展功能,帮助我们在程序启动时获取对应的环境变量
- 国际化消息-拓展功能,帮助我们在做国际化时,快捷的获取不同语言的展示文本
- 事件发布器-拓展功能,帮助我们在容器中进行事件发布和监听
- 生命周期-拓展功能,容器有了对应的生命周期,我们便可以对指定的容器进行销毁和启动
ApplicationContext
的刷新
当我们创建应用容器时,spring会调用应用刷下文的刷新refresh
方法,执行如下流程
- 容器初始化-进行环境变量验证
- 设置容器的启动时间和容器状态
- 加载定义环境变量的验证规则
- 调用自定义验证规则验证环境变量
- bean工厂初始化-创建bean工厂
- 这里主要就是解析bean定义,创建bean工厂
- bean工厂拓展
- 添加了对xml文件中SPEL语言的解析支持
- 添加了对属性编辑器,支持属性类型转换
- 添加了AOP的支持
- 自定义容器-容器bean后置处理器的添加
- 调用容器bean后置处理器
- 执行bean定义注册后处理器
- 执行容器级bean后置处理器
- 注册bean后置处理器
- 注册国际化消息
- 注册事件多播器
- 自定义容器-自定义刷新操作
- 注册事件监听器
- bean初始化
- 添加类型转换器
- 添加嵌入值解析器
- 初始化AOP bean和单例bean
- 容器启动完成
- 加载自定义生命周期,并调用启动方法
- 发布容器完成事件
代码执行的流程如下所示:
总结以下,通过应用容器我们可以实现的拓展功能有:
- 环境变量验证
- bean配置支持SPEL
- 属性编辑器
- 容器级bean后置处理器
- bean后置处理器
- 国际化消息
- 事件的发布与监听
- 生命周期监控
ApplicationContext
的生命周期
容器的启动,暂停,刷新和关闭我们可以调用容器对应的start
、stop
、refresh
、close
方法完成。
容器的暂停和恢复很简单,只调用了自定义生命周期的暂停和恢复方法,并发布相应的事件
容器的暂停代码:
@Override
public void stop() {
getLifecycleProcessor().stop();
publishEvent(new ContextStoppedEvent(this));
}
容器的恢复代码:
@Override
public void start() {
getLifecycleProcessor().start();
publishEvent(new ContextStartedEvent(this));
}
而容器的关闭则需要做如下操作
- 发布关闭事件
- 关闭所有自定义生命周期
- 销毁bean对象
- 关闭bean工厂
- 自定义容器的操作
- 容器状态更新
对应代码则为:
@Override
public void close() {
synchronized (this.startupShutdownMonitor) {
doClose();
// 如果我们注册了一个 JVM 关闭挂钩,我们现在就不再需要它了:我们已经明确地关闭了上下文。
if (this.shutdownHook != null) {
try {
Runtime.getRuntime().removeShutdownHook(this.shutdownHook);
}
catch (IllegalStateException ex) {
//忽略 - 虚拟机已经关闭
}
}
}
}
protected void doClose() {
if (this.active.get() && this.closed.compareAndSet(false, true)) {
if (logger.isDebugEnabled()) {
logger.debug("Closing " + this);
}
LiveBeansView.unregisterApplicationContext(this);
try {
// 发布关机事件。
publishEvent(new ContextClosedEvent(this));
}
catch (Throwable ex) {
logger.warn("Exception thrown from ApplicationListener handling ContextClosedEvent", ex);
}
// 停止所有 Lifecycle bean,以避免在单个销毁过程中出现延迟。
if (this.lifecycleProcessor != null) {
try {
this.lifecycleProcessor.onClose();
}
catch (Throwable ex) {
logger.warn("Exception thrown from LifecycleProcessor on context close", ex);
}
}
// 销毁上下文的 BeanFactory 中所有缓存的单例。
destroyBeans();
// bean工厂置为空
closeBeanFactory();
// 自定义容器的其他关闭后操作
onClose();
//上下文激活状态为关闭
this.active.set(false);
}
}
监听ApplicationContext
状态
我们可以通过监听容器在各个阶段发出的事件来完成对容器状态的监听,当然也可以创建自定义容器来在启动和关闭时执行其他操作。例如:
我们先创建自定义容器实现ClassPathXmlApplicationContext
, 并重写onRefresh
和onClose
方法:
public class MyApplicationContext extends ClassPathXmlApplicationContext {
public MyApplicationContext(String configLocation) throws BeansException {
super(configLocation);
}
@Override
protected void onRefresh() throws BeansException {
super.onRefresh();
System.out.println("容器刷新时,其他处理");
}
@Override
protected void onClose() {
super.onClose();
System.out.println("容器关闭后,其他处理");
}
}
然后创建相应的监听器:
public class MyStartListener implements ApplicationListener<ContextStartedEvent> {
@Override
public void onApplicationEvent(ContextStartedEvent event) {
System.out.println("接收事件"+ event);
}
}
public class MyStopListener implements ApplicationListener<ContextStoppedEvent> {
@Override
public void onApplicationEvent(ContextStoppedEvent event) {
System.out.println("接受事件"+event);
}
}
public class MyRefreshListener implements ApplicationListener<ContextStartedEvent> {
@Override
public void onApplicationEvent(ContextStartedEvent event) {
System.out.println("接受事件"+event);
}
}
public class MyClosedListener implements ApplicationListener<ContextClosedEvent> {
@Override
public void onApplicationEvent(ContextClosedEvent event) {
System.out.println("接收事件" + event);
}
}
我们的xml配置为:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- 自定义一个bean,用于验证容器各个阶段是否可以正常获取到 -->
<bean id="a" class="org.jdbc.demo.A">
<property name="c" value="awqe"/>
<property name="b">
<bean class="org.jdbc.demo.A.B">
<property name="s" value="sds"/>
</bean>
</property>
<property name="date" value="2023-03-24 00:00:00"/>
</bean>
<bean id="myListener" class="org.jdbc.demo.MyListener"/>
<bean id="myClosedListener" class="org.jdbc.demo.MyClosedListener"/>
<bean id="myRefreshListener" class="org.jdbc.demo.MyRefreshListener"/>
<bean id="myStartListener" class="org.jdbc.demo.MyStartListener"/>
<bean id="myStopListener" class="org.jdbc.demo.MyStopListener"/>
</beans>
编写测试类:
public class DemoTest {
public static void main(String[] args) {
MyApplicationContext beanFactory = new MyApplicationContext("spring/application-context.xml");
A a = (A) beanFactory.getBean("a");
System.out.println(a);
beanFactory.stop();
a = (A) beanFactory.getBean("a");
System.out.println(a);
beanFactory.start();
a = (A) beanFactory.getBean("a");
System.out.println(a);
beanFactory.close();
try {
a = (A) beanFactory.getBean("a");
System.out.println(a);
}catch (Exception e){
System.out.println("获取bean异常");
}
beanFactory.refresh();
a = (A) beanFactory.getBean("a");
System.out.println(a);
}
}
//----控制台输出----
//容器刷新时,其他处理
//----当前容器已启动----
//A(c=awqe, b=org.jdbc.demo.A$B@5bcea91b, date=Fri Mar 24 00:00:00 CST 2023)
//----当前容器已关闭----
//接受事件org.springframework.context.event.ContextStoppedEvent[source=org.jdbc.demo.MyApplicationContext@6193b845, started on Sat Apr 22 16:54:04 CST 2023]
//A(c=awqe, b=org.jdbc.demo.A$B@5bcea91b, date=Fri Mar 24 00:00:00 CST 2023)
//----当前容器已启动----
//接受事件org.springframework.context.event.ContextStartedEvent[source=org.jdbc.demo.MyApplicationContext@6193b845, started on Sat Apr 22 16:54:04 CST 2023]
//接收事件org.springframework.context.event.ContextStartedEvent[source=org.jdbc.demo.MyApplicationContext@6193b845, started on Sat Apr 22 16:54:04 CST 2023]
//A(c=awqe, b=org.jdbc.demo.A$B@5bcea91b, date=Fri Mar 24 00:00:00 CST 2023)
//接收事件org.springframework.context.event.ContextClosedEvent[source=org.jdbc.demo.MyApplicationContext@6193b845, started on Sat Apr 22 16:54:04 CST 2023]
//----当前容器已关闭----
//容器关闭后,其他处理
//获取bean异常
//当前加载bean名称为:son
//当前加载bean名称为:son
//容器刷新时,其他处理
//----当前容器已启动----
//A(c=awqe, b=org.jdbc.demo.A$B@7a30d1e6, date=Fri Mar 24 00:00:00 CST 2023)
//
//进程已结束,退出代码0
由此我们可以看到,容器的暂停和恢复,并不影响bean的获取,只会影响我们自定义周期的调用
评论区