Spring IOC源码流程解析

    Spring是一个轻量级的java开发框架,主要是简化java工程师的开发过程,其主要的模块是IOC和AOP。何为IOC?Inversion of control,控制反转。什么是控制?控制为什么需要反转?这些都是我们要首先搞清楚的问题。打个比方,我们的JVM中有许多个对象,有些对象是程序运行到一定阶段生成的,有些则是在一开始就需要被创建的,比如WEB当中的Controller, Service, Dao对象。所以我们希望的最好有一个统筹全局的“容器”替我们管理这些对象,而并非要我们将这些对象的创建控制写死在业务程序中,最好是将它们的创建与否交给配置文件或者是配置类,与正常的业务逻辑解耦。所以伟大的IOC模块就诞生了!

    市面上讲解Spring的书都各有千秋,有教你如何使用的Spring in action。也有教你阅读Spring源码的Spring 技术内幕。作为java新手,如果止步于使用Spring,自然第一本书或者几篇博客足矣。但是,正所谓判断一个程序员的能力并不是其写代码的能力,而是解决问题的能力。比如 阿里巴巴开源出来的RPC框架dubbo就是和Spring无缝结合,在配置dubbo的过程很多时候会抛出一些Spring的异常,这时候就能体现出了解Spring流程细节的优势。本人也正是出于这样的目的所以才选择了着手于Spring的源码。

    但是,Spring的控制流程过于复杂,涵盖的点比较多,时间久了,非常容易遗忘。所以本人出于化繁为简的目的,梳理下IOC的主要流程,写下这篇日志。

    正所谓不识庐山真面目,只缘身在此山中。IOC源码阅读的难度也在于一开始我们就陷入到其中的一条支线结果导致我们忘记了IOC的目的所在。我们应该一开始就有个大局观,就如同IOC的目的无外乎3种:

1. Resource定位,也就我们所说的配置文件(Xml),配置类(JavaConfig)。必须先找到描述bean对象的文件,才好完成后面对象创建与管理。

2.BeanDefinition的解析和注册,承继上面的找到bean对象描述信息之后,我们需要在内存中用命为BeanDefinition的对象去封装它。何谓注册?顾名思义,注册就是为了后面的查询服务的,我们前文不是提及过希望有一个“容器”去管理它们吗。所以注册就是以beanName 为key,beanDefinition为value注册到一个concurrentHashMap中去。

3.Ioc的依赖注入,说了那么多,肯定有一个获取bean的方法,没错我们可以通过getbean()的方式获取bean对象,而依赖注入就是在这个方法内部完成的,内部是以递归的方式完成的。所以当我们在开发时候碰到空指针异常的时候,大多数时候是因为我们Spring 配置文件处理不当,bean与bean之间的依赖关系没处理好

下面我将以我们平时使用Spring 的API为入口,挖出IOC的所有行为。

ClassPathXmlApplicationContext classPathXmlApplicationContext =new ClassPathXmlApplicationContext("spring.xml");

Person scientist = (Person) classPathXmlApplicationContext.getBean("Scientist");

这是我们平时写测试用例,初始化Spring容器和获得bean的惯用手段。那么初始化阶段究竟干了什么事呢?

初始化beanFactory

    说了这么多遍beanDefinion,是时候让大家见识一下beanDefinition的真面目了。其实如果熟练使用Spring IOC的同学自己都可以根据Xml中的标签设计出来了!

beanDefinition的结构

    以上完成了beanDefinition的解析和注册,下面我们来看看它是如何完成依赖注入的,上文我已经解释过bean的实例化和依赖注入是在getBean()的过程中完成的!


bean的实例化外加依赖注入

        如此,整个基于XML的依赖注入过程已经解释清楚了。真正理解这个过程的朋友也就能明白为什么当bean之间存在循环依赖的时候,Spring会爆BeanCurrentlyInCreationException异常,设想一下现在A依赖于B,B依赖于A。那样就会在getBean(A)的时候,发现要去getBean(B),发现bean池中没有B的bean,就会去创建B的bean,但是在创建的过程中发现有需要创建A,就去调getBean(A)。。。如此一来,bean池中既没有A的bean,也没有B的bean,所以Spring会去检测这样的情况,他的检测方法也比较简单,只要为该bean打一个标记,当一个依赖循环过来的时候,发现“正在创建中!”,即可判断存在循环依赖,抛出异常!

       真正的依赖注入过程要比这个复杂的多得多,这是我只是为了我们更好的使用IOC才取其精华,以便更好的理解整个流程。

    当然这里也要提及一句,IOC发展至今玩法也是多种多样的,光依赖注入的方式就有三种,而不再是仅仅的Xml,但是不管怎么变,最基本的实例化方式和管理方式依旧万变不离其宗!

    补充一下Spring bean的生命周期


bean的生命周期

    上面我只说了实例化bean和依赖注入,这是比较基本的功能。

     后续Spring还会检查bean有没有实现Aware接口,这是bean对beanFactory的感知功能IOC的一个特性,其实就是将beanFactory注入到该bean中,这样该bean本身也能使用IOC容器。

      bean的前置处理postProcessBeforeInitialzation,这个函数会先于InitialzationBean执行,因此称为前置处理。 所有Aware接口的注入就是在这一步完成的。

      InitializingBean,是一个接口,内部有一个函数afterPropertiesSet()。Spring为了降低对客户代码的侵入性,最好让bean的一些初始化行为在容器的创建过程中也直接完成,所以Spring会在初始化的过程中判断该bean有没有实现InitializingBean接口,如果实现了就是调用afterPropertiesSet里的方法,也可以给bean的配置提供了init-method属性,init-method本质上仍然使用了InitializingBean接口。 

     bean的后置处理postProcessAfterInitialzation,AOP就是在这个阶段完成的,可见我AOP文章。

     DisposableBean,是一个接口,和initial一样,内部有一个destory方法,通过给destroy-method指定函数。由用户在业务逻辑中调用,完成销毁。

    码农和工程师的区别,将会砥砺我一直刨根问底下去!

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

推荐阅读更多精彩内容