Spring揭秘-IOC容器的原理

Bean是怎么创建的?

Magic?

上图中Magic Happens Here实现可以分为两个阶段,容器启动阶段、Bean实例化阶段。

容器启动阶段

容器依赖BeanDefinitionReader对加载的Configuration Metadata进行解析,将得到的信息保存为BeanDefinition,注册到BeanDefinitionRegistry后,容器的启动阶段就完成了。

BeanFactoryPostProcessor

Spring提供了一种叫做 BeanFactoryPostProcessor 的容器扩展机制,它允许我们在容器启动阶段的最后修改BeanDefinition,达到修改Bean属性的目的,
Spring提供了几个现成的 BeanFactoryPostProcessor 实现类

  • PropertyPlaceholderConfigurer
    我们配置DataSource时通常会用${jdbc.url}等占位符,容器启动阶段完成后,BeanDefinition中仍存在占位符,当PropertyPlaceholderConfigurer 作为 BeanFactoryPostProcessor 被应用时,它会使用properties或yml配置文件中的配置信息来替换相应 BeanDefinition 中占位符所表示的属性值。
  • PropertyOverrideConfigurer
    它的properties文件中的配置项覆盖掉了原来XML中的bean定义的property信息,并且对于Bean是透明的。
  • CustomEditorConfigurer
    配置文件的property都是 String 类型,但在程序中我们可能需要各种类型的对象,比如Date,CustomEditorConfigurer能帮我们完成这种类型的转换。

Bean实例化阶段

经过容器启动阶段,现在所有的bean定义信息都通过 BeanDefinition 的方式注册到了 BeanDefinitionRegistry 中。当某个请求方通过容器的 getBean 方法明确地请求某个对象,或者因依赖关系容器需要隐式地调用 getBean 方法时,就会触发Bean实例化阶段。Spring容器将对其所管理的对象全部给予统一的生命周期管理,这些被管理的对象完全摆脱了原来那种new完后被使用,脱离作用域后即被回收的命运。
下图是实例化的过程。


Bean的实例化过程
  1. 实例化对象
    容器内部采用策略模式决定用什么方式初始化bean,通常通过反射或者cglib初始化bean或者动态生成子类,默认情况下,容器内部采用的是CglibSubclassingInstantiationStrategy,它不会在对象初始化完成后直接返回,而是以 BeanWrapper 对构造完成的对象实例进行包装,返回相应的 BeanWrapper 实例。
  2. 设置对象属性
    BeanWrapper继承了PropertyAccessor接口,可以以统一的方式对对象属性进行访问。
  3. 检查Aware相关接口并设置相关依赖
    当对象实例化完成并且相关属性以及依赖设置完成之后,Spring容器会检查当前对象实例是否实现了一系列的以 Aware 命名结尾的接口定义。如果是,则将这些 Aware 接口定义中规定的依赖注入给当前对象实例。
    • 针对BeanFactory容器,BeanNameAware、BeanClassLoaderAware、BeanFactoryAware这几个接口,分别会将beanName、ClassLoader、BeanFactory注入到当前对象实例。
    • 针对ApplicationContext容器,ResourceLoaderAware、ApplicationEventPublisherAware、MessageSourceAware、ApplicationContextAware会将自身(aplicationContext)注入到当前对象实例。
  4. BeanPostProcessor前置处理
public interface BeanPostProcessor {
    //前置处理
    Object postProcessBeforeInitialization(Object var1, String var2) throws BeansException;
    //后置处理
    Object postProcessAfterInitialization(Object var1, String var2) throws BeansException;
}

如图,BeanPostProcessor 接受当前对象实例,它可以对当前对象做任何操作,一般用于处理标记接口实现类,或者为当前对象提供代理实现(AOP)。针对 ApplicationContext 容器检查Aware相关接口就是使用之前注册到容器的
ApplicationContextAwareProcessor 这个实现类的前置处理实现的。

  1. 检查是否是InitializingBean
public interface InitializingBean {
    void afterPropertiesSet() throws Exception;
}

在对象实例化过程调用过 BeanPostProcessor 的前置处理之后,会接着检测当前对象是否实现了 InitializingBean 接口,如果是,则会调用其 afterPropertiesSet() 方法进一步调整对象实例的状态,比如在有些情况下,某个业务对象实例化完成后,还不能处于可以使用状态。这个时候就可以让该业务对象实现该接口,并在方法 afterPropertiesSet()中完成对该业务对象的后续处理。

  1. 检查是否配置有自定义的init-method
    实现InitializingBean,Spring容器会有较强的侵入性,所以Spring还提供了另一种方式来指定自定义的对象初始化操作,那就是在XML配置的时候使用 <bean> 的 init-method 属性。

  2. BeanPostProcessor后置处理
    发生在初始化操作后,调用同前置操作。注意,ApplicationContext会自动检测在配置文件中实现了BeanPostProcessor接口的所有bean并把他们注册,BeanFactory需要写代码调用。

  3. DisposableBean 与 destroy-method,注册必要的Destruction相关回调接口
    当所有的一切,该设置的设置,该注入的注入,该调用的调用完成之后,容器将检查singleton类型的bean实例,看其是否实现了 DisposableBean 接口。或者其对应的bean定义是否通过 <bean> 的 destroy-method 属性指定了自定义的对象销毁方法。如果是就会为该实例注册一个用于对象销毁的回调(Callback),以便在这些singleton类型的对象实例销毁之前,执行销毁逻辑。但是销毁逻辑不会自动执行。

    • 对于BeanFactory
      我们必须手动调用 ConfigurableBeanFactory 提供的 destroySingletons() 方法销毁容器中管理的所有singleton类型的对象实例。
    • 对于ApplicationContext
      AbstractApplicationContext 为我们提供了 registerShutdownHook() 方法,该方法底层使用标准的 Runtime 类的 addShutdownHook() 方式来调用相应bean对象的销毁逻辑,从而保证在Java虚拟机退出之前,这些singtleton类型的bean对象实例的自定义销毁逻辑会被执行。

    不过,所有这些规则不包含prototype类型的bean实例,因为prototype
    对象实例在容器实例化并返回给请求方之后,容器就不再管理这种类型对象实例的生命周期了。

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

推荐阅读更多精彩内容

  • Spring框架是一个开源的java平台,基于Spring框架开发服务项目非常容易,迅速和健壮。框架简单理解就是封...
    maomaohbu阅读 554评论 0 1
  • 初始化示例 我们先来看下spring如何手动初始化一个对象 spring源码解析 所以我们先从DefaultLis...
    AbsurdOS阅读 2,451评论 0 2
  • 在Spring中有三种实例化Bean的方式,但是在实际中使用最多的是第一种,其他两种几乎不使用。 实体bean,下...
    mm_cuckoo阅读 334评论 0 1
  • 阿玲早早就回到了家,想起白天发生的一切,不禁发愁。但阿玲也不是个悲观的人,于是便洗洗睡了。 这会子宇轩在自家房里正...
    雅妍阅读 256评论 0 2
  • 外面一片艳阳天 春暖花开 鸟语花香 我的心 却坠落到一个冰窖 彻骨地冰冷 你又一次毫无征兆地消失了 没有只字片语 ...
    蓝梦奇阅读 531评论 11 16