对于Java开发者来讲,Spring无疑是最经常使用也是最基础的框架之一。(此处省略1w字吹Spring)。相信不少同行跟我同样,只是停留在会用的阶段,好比用@Component写一个组件、用@Autowired注入其余组件等等,可是不知道为何能够这么作,Spring是怎么实现的。为了了解这些,我阅读了《Spring源码深度解析》,这本书讲的很详细,可是由于步骤多而复杂容易记混,我就作了一下梳理,先呈现大体流程,但对每一个步骤进行详细描述。数组
概念上的东西仍是要提一嘴的:
Spring用IoC容器来管理Bean。
BeanFactory和ApplicationContext是SpringIoC容器的两种表现形式。
BeanFactory定义了简单IoC容器的基本功能。
ApplicationContext实现了BeanFactory,且经过继承MessageSource、ResourceLoader、ApplicationEventPublisher接口,添加了许多高级容器的特性。框架
这里以XmlBeanFactory为表明,看容器是怎么工做的。
新建一个XmlBeanFactory很简单,只须要你有一个符合格式的xml文件,里面用<bean>标签设置你但愿被容器加载的Bean:函数
BeanFactory bf = new XmlBeanFactory(new ClassPathResource("test.xml"))
新建了一个ClassPathResource资源对象做为参数传入XmlBeanFactory的构造函数。
点进XmlBeanFactory类,XmlBeanFactory会先调用父类构造器,一直跟踪到AbstractAutowireCapableBeanFactory,会看到调用了三次ignoreDependencyInterface用来忽略给定接口的自动装配功能(后面会提到);再调用this.reader.loadBeanDefinitions(Resource),用本身持有的XmlBeanDefinitionReader解析传入的资源。因此最宏观的三个步骤:
一、新建了一个ClassPathResource资源。抽象出一个资源类来表示资源;
二、调用了ignoreDependencyInterface忽略指定接口的自动装配功能;
三、委托XmlBeanDefinitionReader解析资源。
重点确定在第三步了,点进XmlBeanDefinitionReader,到loadBeanDefinitions(EncodedResource),先从
Resource获取InputStream构形成InputSource,做为参数调用doLoadBeanDefinitions(InputSource, Resource)开始真正的解析。因此第三步下面是两个小步骤:
3.一、从Resource获取输入流;
3.二、调用doLoadBeanDefinitions继续解析。
再看doLoadBeanDefinitions方法,主要是两个方法,doLoadDocument(inputSource, resource)解析输入流返回一个Document对象;registerBeanDefinitions(doc, resource)继续解析Document返回解析的Bean数量,因此3.2下面是两个步骤:
3.2.一、将资源解析成Document对象(这个步骤这边就不展开了,有兴自究);
3.2.二、解析Document,提取注册Bean。
来到registerBeanDefinitions方法,这里建立了一个BeanDefinitionDocumentReader对象负责具体解析(是否是以为又冒出了不认识的类,框架就是这样,遵循单一职责的原则,把一个集中的逻辑放到其余类中处理),它调用doRegisterBeanDefinitions(doc.getDocumentElement()),提取Document的root做为参数继续解析,先查找解析profile属性,将表示环境的属性注册到Environment;而后遍历root每一个子节点,若是是默认标签,调用parseDefaultElement进行解析;若是是自定义标签,就调用delegate.parseCustomElement。因此3.2.2下面是这几个步骤:
3.2.2.一、建立BeanDefinitionDocumentReader委托解析对象;
3.2.2.二、解析profile属性到Environment;
3.2.2.三、遍历子节点,继续解析默认标签和自定义标签。
咱们这边主要分析默认标签的解析,自定义的有兴自究。首先根据标签类型选择不一样的处理方法,类型分别是import、alias、bean和beans。重点确定是对bean标签的解析,进入processBeanDefinition方法,咱们看到里面先委托BeanDefinitionParserDelegate解析出一个持有bean信息的BeanDefinitionHolder;若是BeanDefinitionHolder不为空且子节点下存在自定义标签,再解析它们;而后对解析完成后的BeanDefinitionHolder进行注册,注册过程很简单就是将BeanDefinitionHolder持有的beanName和BeanDefinition的键值对、beanName和每一个alias别名的键值对保存在容器中;最后发出bean已注册完成的事件通知,因此这里分为4步:
3.2.2.3.一、委托BeanDefinitionParserDelegate解析返回BeanDefinitionHolder;
3.2.2.3.二、解析存在的自定义标签;
3.2.2.3.三、解析完成后注册;
3.2.2.3.四、发出响应事件。
到了这里终于开始具体的解析,过程其实就是先解析出beanName、alias别名,而后把其他各类标签,如class、scope、lazy-init等属性解析成用于属性承载的BeanDefinition对象的成员变量,最后将beanName、alias数组和BeanDefinition封装成BeanDefinitionHolder对象返回。具体各类属性的功能和规则这边就不展开了,有兴自究。
对于import、alias和beans标签,简述一下:alias的解析和bean中的alias解析差很少,也是将每个别名与beanName以map形式保存;impot能够导入其余配置文件,解析过程就是找到那个文件而后递归进行解析;beans就是把多个bean标签包起来,而后遍历解析每个bean标签。this
画一个流程图做为总结:spa