我是靠谱客的博主 阔达麦片,最近开发中收集的这篇文章主要介绍ApplicationContext探究ApplicationContext探究,觉得挺不错的,现在分享给大家,希望可以做个参考。

概述

ApplicationContext探究

ApplicationContext类结构树

上图表示了ApplicationContext的依赖关系。
BeanFactory是Spring容器依赖注入的基础。位于类结构树的顶端,接口中最重要的方法就是getBean(String)。得到特定名称的Bean对象。
ListableBeanFactory是用来访问容器内bean的相关信息,根据方法名可以看出,就相当于对一个容器中的内容相关信息的获取。
HierarchicalBeanFactory是容器层次化。获取父级BeanFactory。但是在ApplicationContext接口中没有setParent()方法,在ConfigurableApplicationContext接口中出现了set方法。这个在后面说到的时候再说。通过bean名称判断当前容器是否包含这个bean实例。
ResourceLoader是资源加载接口,用于对不同的Resource进行加载。Resource getResource(String location)通过一个标识加载资源信息。
ResourcePatternResolverResourceLoader的扩展,将一个标识拆分成多个资源。
MessageSource接口,�提供了消息处理的功能。在web项目上用于国际化。
EnvironmentCapable运行环境接口。通过这个接口可以获取到Environment。spring的Environment
ApplicationEventPublisher提供了发布event组件的接口。spring的事件体系 和Spring内置事件处理

ApplicationContext中的主要实现类解读

ConfigurableApplicationContext接口方法完成了对ApplicationContext进一步的扩展。ApplicationContext以及他继承的部分接口都是获取相关PropertiesConfigurableApplicationContext中提供了set方法。
setId对应ApplicationContext中的get方法。
setParent对应HierarchicalBeanFactory中的get方法。
setEnvironment对应EnvironmentCapable中的get方法。
addApplicationListener对应ApplicationEventPublisher。扩展之后可以发布事件,也有了监听能力。
同时重新定义了getEnvironment()
isActive(),refresh()close()方法的定义让实现这个接口的类具有了状态标识。
addBeanFactoryPostProcessor为操作容器内的bean提供了接口。BeanFactoryPostProcessor接口,允许修改容器中的定义的bean
以上就是ConfigurableApplicationContext接口对ApplicationContext的扩展内容。

AbstractApplicationContext 抽象类,这个应该是Spring中所有ApplicationContext的父类了,实现了一些比较核心的方法。

先看一下AbstractApplicationContext的类结构树。

AbstractApplicationContext

AbstractApplicationContext抽象类。
首先他定义了一些常量:
MESSAGE_SOURCE_BEAN_NAME 统一的信息资源名称, messageSource
LIFECYCLE_PROCESSOR_BEAN_NAME 统一的生命周期处理工具Bean的名称, lifecycleProcessor
APPLICATION_EVENT_MULTICASTER_BEAN_NAME 上下文事件的多路广播名称, applicationEventMulticaster
然后定义了一些 Properties,在接口中都有相关的set和get方法。
id=(String)displayName,ID context的唯一标识,displayName和ID的生成方法相同。
(ApplicationContext)parent,父级Context。
(long)startupDate,启动时间,上下文开始运行的系统毫秒级时间
(AtomicBoolean)active,是否是活跃的。
(AtomicBoolean)closed,是否关闭
(List<BeanFactoryPostProcessor>)beanFactoryPostProcessors,bean工厂的后处理器,用来申请refresh服务。
(Object)startupShutdownMonitor, refresh和destroy监视器,
(Thread)shutdownHook,
(ResourcePatternResolver)resourcePatternResolver资源加载器,
(LifecycleProcessor)lifecycleProcessor管理 当前context中bean的生命周期。
(MessageSource)messageSource,将MessageSource委托到context中。
(ApplicationEventMulticaster)applicationEventMulticaster 事件发布时的辅助类。
(Set<ApplicationListener<?>>)applicationListeners 监听器,LinkedHashSet。
(ConfigurableEnvironment)environment。初始化环境参数。
同时他还实现了 DisposableBean接口,提供了对单例的 bean进行销毁的操作方法。
简单梳理一下 AbstractApplicationContext在初始化时执行的操作,因为基本上这就是 ApplicationContext的最根本的父类了,所以在初始化容器的时候,都会想执行他的静态块和构造方法。最先执行的应该是他的静态块语句 ContextClosedEvent.class.getName();

静态块的注释为:为了避免应用程序在weblogic8.1关闭的时候出现加载类加载异常的问题,所以较早的加载这个ContextClosedEvent事件。 暂时没有发现是干什么用的。

然后实例化AbstractApplicationContext对象。无参构造方法是给自己的resourcePatternResolver属性完成赋值。有参构造方法是设置了父类parent属性。 作为一个父类,本身实现了一些方法,这些方法在后面用到的时候详细解读。refresh()方法。

AbstractRefreshableApplicationContext 继承了AbstractApplicationContext,实现了部分抽象方法。扩展了一些Properties

(Boolean)allowBeanDefinitionOverriding 允许对定义的Bean重写。
(Boolean)allowCircularReferences 允许循环引用。
(DefaultListableBeanFactory)beanFactory 容器内的beanFactory。
final (Object)beanFactoryMonitor beanFactory监控。
1.实现了AbstractApplicationContextrefreshBeanFactory方法 完成了对容器底层的beanFactory的刷新。这个方法的流程需要详细解读。比较重要。

判断是否存在BeanFactory,销毁之前的bean,关闭之前的BeanFactory。重新实例化BeanFactory 完成相关BeanFactory的配置,执行loadBeanDefinitions抽象方法。加载bean的定义应该是一个很关键的操作。

2.重写了cancelRefresh方法。
3.实现了closeBeanFactory方法。
4.自定义了createBeanFactory方法。
5.定义了loadBeanDefinitions抽象方法。
同时还有定义Properties的相关set方法。
AbstractRefreshableApplicationContext类与DefaultListableBeanFactory类有很大关系。后面要介绍。这个BeanFactory作为了容器内部的BeanFactory。

慢慢整理的过程中就明白了,虽然ApplicationContext继承了BeanFactory,但是他自己不仅仅是一个bean实例对象的提供者,相关功能有内部的BeanFactory实现。ApplicationContext更专注容器的功能和控制功能。这个应该算是代理模式

AbstractRefreshableConfigApplicationContextAbstractRefreshableApplicationContext进行扩展。同时继承了BeanNameAwareInitializingBean

本身扩展了多个配置文件处理方法。
1.定义了一个setIdCalled属性,用于判断是否调用过setId方法。
2.(String[])configLocations相关配置文件
重写了setId方法,继承了BeanNameAware,实现了setBeanName方法。这个意思是将context本身当做一个bean,设置一个beanName。但是在设置这个beanName有一个条件就是没有调用setId方法,这也就是setIdCalled属性的意义。
继承了InitializingBean 实现了afterPropertiesSet方法,方法的意思是定义初始化方法的方式。如果不是active的话就执行了refresh()方法。refresh方法的实现是AbstractApplicationContext

AbstractRefreshableConfigApplicationContext扩展后就将ApplicationContext也定义成了一个bean。执行一些与bean相关的方法。

AbstractRefreshableConfigApplicationContext 有两个抽象类的扩展AbstractXmlApplicationContextAbstractRefreshableWebApplicationContext

从简单的开始AbstractXmlApplicationContext

定义了一个validating属性,默认值为true,提供set方法。
实现了loadBeanDefinitions方法,通过XmlBeanDefinitionReader加载bean的定义。
就是实现了这个loadBeanDefinitions方法,完成了bean的定义。
ClassPathXmlApplicationContext继承了AbstractXmlApplicationContext。实现了AbstractXmlApplicationContext的抽象方法。定义了(Resource[])configResources属性。通过构造方法来完成对参数的赋值。主要的构造方法:

public ClassPathXmlApplicationContext(String[] configLocations, boolean refresh, ApplicationContext parent)throws BeansException {
        super(parent);
        setConfigLocations(configLocations);
        if (refresh) {
            refresh();
        }
    }

这个构造方法相对来说用的比较普遍。接下来就是使用容器时初始化的过程了。
FileSystemXmlApplicationContextClassPathXmlApplicationContext相比也就是加载configLocations的方法不一样。一个是classPath路径,一个是文件系统的路径。

使用ClassPathXmlApplicationContext时加载过程顺序。
DefaultResourceLoader -> AbstractApplicationContext -> AbstractRefreshableApplicationContext-> AbstractRefreshableConfigApplicationContext -> AbstractXmlApplicationContext-> ClassPathXmlApplicationContext。在实例化ClassPathXmlApplicationContext过程中父类都会被加载,实例化,上面是父类的加载顺序。

在实例化 ClassPathXmlApplicationContext 过程中发什么了什么?
最先执行的应该是静态块
1.AbstractApplicationContext的静态块
2.执行ClassPathXmlApplicationContext的构造方法
    1.进入到父类构造方法->DefaultResourceLoader->获取ClassLoader。
    2.AbstractApplicationContext的相关属性赋值,进入构造方法->获取ResourcePatternResolver。(具体是实例化了一个PathMatchingResourcePatternResolver对象)
    3.AbstractRefreshableApplicationContext相关属性赋值
    4.AbstractRefreshableConfigApplicationContext相关属性赋值
    5.AbstractXmlApplicationContext相关属性赋值。
    前面的一段就是super()方法的操作。
执行setConfigLocations(configLocations);方法实现在 AbstractRefreshableConfigApplicationContext。
将configLocations赋值到了AbstractRefreshableConfigApplicationContext中的configLocations属性中。

refresh();refresh <span class="hljs-keyword">true</span>是必须的 这个方法必须执行,在初始化容器的时候。
refresh是关键方法
    prepareRefresh();
        <span class="hljs-number">1</span>.给startupDate和active赋值
        <span class="hljs-number">2</span>.日志
        <span class="hljs-number">3</span>.initPropertySources()方法 初始化在这个上下文环境中所有占位符的属性资源。自己实现的是一个空的方法。 但是在web方向中这个方法被重写过。
        <span class="hljs-number">4</span>.getEnvironment().validateRequiredProperties();
            <span class="hljs-number">1</span>.首先是getEnvironment()-&gt;createEnvironment() 实例化了一个StandardEnvironment();
            <span class="hljs-number">2</span>.执行了validateRequiredProperties方法。有AbstractEnvironment实现的方法。具体的环境方面 单独分析。
            
    ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
        <span class="hljs-number">1</span>.获取刷新后的beanFactory。
            <span class="hljs-number">1</span>.refreshBeanFactory-&gt;抽象方法,具体的实现在AbstractRefreshableApplicationContext中。
                <span class="hljs-number">1</span>.判断是否用BeanFactory。有:销毁bean,关闭BeanFactory。初始化的时候是没有BeanFactory的。
                <span class="hljs-number">2</span>.创建一个BeanFactory,createBeanFactory();DefaultListableBeanFactory作为整个context的内部工厂。
                <span class="hljs-number">3</span>.给BeanFactory设置一个SerializationId,
                <span class="hljs-number">4</span>.customizeBeanFactory定制BeanFactory 允许循环引用和允许对定义的bean重写。默认为<span class="hljs-keyword">null</span>。
                <span class="hljs-number">5</span>.关键部分loadBeanDefinitions加载bean的定义。抽象方法,AbstractXmlApplicationContext 有实现。
                    <span class="hljs-number">1</span>.根据上一步实例化的BeanFactory来实例化一个 XmlBeanDefinitionReader
                    <span class="hljs-number">2</span>.给XmlBeanDefinitionReader设置Environment,这个Environment应该在之前创建过。
                    <span class="hljs-number">3</span>.给XmlBeanDefinitionReader设置ResourceLoader。
                    <span class="hljs-number">4</span>.给XmlBeanDefinitionReader设置ResourceEntityResolver。
                    <span class="hljs-number">5</span>.initBeanDefinitionReader(beanDefinitionReader);
                        <span class="hljs-number">1</span>.设置validating。默认是为<span class="hljs-keyword">true</span>。就是对beanDefinitionReader进行初始化操作。
                    <span class="hljs-number">6</span>.loadBeanDefinitions(beanDefinitionReader);
                        设置ConfigLocations。可能是Resource[]也可能是String[]。
        refreshBeanFactory方法是为了让context中的BeanFactory赋值,其中包括factory中的XmlBeanDefinitionReader的操作。 
        ApplicationContext中的factory被定义在AbstractRefreshableApplicationContext。
        <span class="hljs-number">2</span>.ConfigurableListableBeanFactory beanFactory = getBeanFactory();通过BeanFactoryMonitor校验得到BeanFactory。

    prepareBeanFactory(beanFactory);
        对ApplicationContext内部的BeanFactory进行操作、赋值。
    
    postProcessBeanFactory(beanFactory);
        bean的Post-Process操作。

    invokeBeanFactoryPostProcessors(beanFactory);
        实例化和调用注册为PostProcess的bean。

    registerBeanPostProcessors(beanFactory);

    initMessageSource();

    initApplicationEventMulticaster();

    onRefresh();

    registerListeners();

    finishBeanFactoryInitialization(beanFactory);

    finishRefresh();

整个过程就是ApplicationContext实例化时的操作。有部分说的不是很清楚的是因为我自己还没有去看如何实现的,涉及到DefaultListableBeanFactoryXmlBeanDefinitionReader。一个是ApplicationContext的内部BeanFactory和定义Bean的加载器(不知道用加载器对不对)。
这里暂时没有涉及到web项目中的WebApplicationContext

最后

以上就是阔达麦片为你收集整理的ApplicationContext探究ApplicationContext探究的全部内容,希望文章能够帮你解决ApplicationContext探究ApplicationContext探究所遇到的程序开发问题。

如果觉得靠谱客网站的内容还不错,欢迎将靠谱客网站推荐给程序员好友。

本图文内容来源于网友提供,作为学习参考使用,或来自网络收集整理,版权属于原作者所有。
点赞(69)

评论列表共有 0 条评论

立即
投稿
返回
顶部