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

spring 版本为 4.1.6。 个人见解,可能有很多不对的地方,请帮忙指出,谢谢!

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容

  • Spring容器高层视图 Spring 启动时读取应用程序提供的Bean配置信息,并在Spring容器中生成一份相...
    Theriseof阅读 2,795评论 1 24
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,567评论 18 399
  • 什么是Spring Spring是一个开源的Java EE开发框架。Spring框架的核心功能可以应用在任何Jav...
    jemmm阅读 16,438评论 1 133
  • sdfdsafg
    66CQcq66阅读 162评论 0 0
  • 清晨曦光盼日出, 单手托月青海湖。 海军基地隐薄雾, 月上塔尖现美图。
    高原_4697阅读 281评论 2 4