Spring源码分析之IOC容器(一)
Spring作为当今风靡世界的Web领域的第一框架,作为一名Java开发程序员是一定要掌握的,除了需要掌握基本的使用之外,更需要掌握其实现原理,因为我们往往在开发的过程中,会出现各种各样的异常问题。而且这样的问题去百度有时候往往也找不到特别有效的解决方法,因为问题的原因非常多而百度的又不准确,这个时候怎么办呢?在熟练掌握Spring代码的情况下,我们可以根据提示的异常信息,去跟到源代码的地方,可以准确的定位到异常的所在,这就犹如debug调试自己的业务逻辑一样轻松了,所以Spring的源代码非常值得一读。还有另外一个好处就是,Spring的构建非常的复杂,但同时它的设计也是非常的灵性的,不仅对我们自身的软件设计能力会有很大的提高,感受到软件设计的魅力所在。其实我本来是想等我把Spring的代码完全读透之后再写下这些博文,但是感觉写博客就相当于对自己的一个总结,可能边分析边学下来比较好,下面,就让我来一点一点分析Spring的源码。
Spring家族的构成非常的庞大,网上有很多图,我也随手拎了一张过来,如下图所示:
通过这张图,我们可以看到处于最下面的是Beans、Core、Context和SpEl四个模块,SpEl是Spring的一个表达式,并没有那么重要,所以我们暂时就不去分析这个了。重点讲一讲其他的三个模块。我借用淘宝大神许令波在《深入分析Web技术内幕》一书中的比喻,Beans相当于Java的对象,而Java对象封装了数据,一个IOC容器中往往会有很多的Bean,我们在这里可以将Bean比喻成演员。Context就相当于容器,给演员,也就是Bean提供了一个舞台,那么最后演员有了,舞台也有了,每个演员不可能只会自己演自己的,也不可能没有道具和服装,所以这个时候Core组件排上用场了,按照这么说的话,Core组件可以维护每个Bean之间的关系,也就相当于这些演员在舞台上要演出的“道具”和“服装”等等。如果这个比喻你还不能理解的话,可以再打一个比喻。三国演义中,每个主公都有自己的武将,可以把武将就看作一个Bean,主公作为一城之主,当敌人来犯的时候必须调配武将,给他们分配兵马,而且打仗的时候需要各个武将之间的配合方能取胜,这个时候我们就可以把主公当作一个容器,主公分发的兵马,粮草器械等等是Core组件,这个用来维持每个武将之间的关系。这三者之间的关系大概就是这样了,只要理解了其实很多情景都可以用IOC来解释了。今天我们就主要说说IIOC三大组件之一的Bean组件。
首先我们要知道Bean组件是怎么设计的,有哪些类和接口定义了这些功能,画出UML图来表示这些结构和组成。Bean组件位于Spring的org.springframework.beans包下面,如图所示:
这个包其实解决了三件事情:Bean的定义、Bean的创建和Bean的解析。而我们在使用Spring框架的时候要做的就是完成Bean的创建,也就是编写配置文件。Spring的Bean组件的实现完全上是一个工厂模式,位于最顶层的是一个名叫BeanFactory的接口,IOC功能的Bean组件的类结构如下图所示:
最终的实现类是DefaultListableBeanFactory,它实现了所有的接口和类。最顶层的类是BeanFactory,从字面的意思可以理解为bean工厂,是bean组成的最基本的类。它有三个子接口:ListableBeanFactory、HierarchicalBeanFactory和AutowireCapableBeanFactory三个接口。这三个接口的功能肯定是不一样的,分别将代码贴出来。
ListableBeanFactory:这个接口的Bean工厂表示里面的bean元素是可列表的
package org.springframework.beans.factory; import java.lang.annotation.Annotation;
import java.util.Map; import org.springframework.beans.BeansException; public interface ListableBeanFactory extends BeanFactory { boolean containsBeanDefinition(String beanName);
int getBeanDefinitionCount(); String[] getBeanDefinitionNames();
String[] getBeanNamesForType(Class<?> type);
String[] getBeanNamesForType(Class<?> type, boolean includeNonSingletons, boolean allowEagerInit); <T> Map<String, T> getBeansOfType(Class<T> type) throws BeansException; <T> Map<String, T> getBeansOfType(Class<T> type, boolean includeNonSingletons, boolean allowEagerInit)
throws BeansException; Map<String, Object> getBeansWithAnnotation(Class<? extends Annotation> annotationType)
throws BeansException; <A extends Annotation> A findAnnotationOnBean(String beanName, Class<A> annotationType);
}
HierarchicalBeanFactory:这个接口的Bean表示这些Bean是有继承关系的。
package org.springframework.beans.factory; public interface HierarchicalBeanFactory extends BeanFactory { BeanFactory getParentBeanFactory(); boolean containsLocalBean(String name); }
AutowireCapableBeanFactory:表示Bean是可以自动装配的。
package org.springframework.beans.factory.config; import java.util.Set; import org.springframework.beans.BeansException;
import org.springframework.beans.TypeConverter;
import org.springframework.beans.factory.BeanFactory; public interface AutowireCapableBeanFactory extends BeanFactory { int AUTOWIRE_NO = 0; int AUTOWIRE_BY_NAME = 1; int AUTOWIRE_BY_TYPE = 2; int AUTOWIRE_CONSTRUCTOR = 3; @Deprecated
int AUTOWIRE_AUTODETECT = 4; //-------------------------------------------------------------------------
// Typical methods for creating and populating external bean instances
//------------------------------------------------------------------------- <T> T createBean(Class<T> beanClass) throws BeansException; void autowireBean(Object existingBean) throws BeansException; Object configureBean(Object existingBean, String beanName) throws BeansException; Object resolveDependency(DependencyDescriptor descriptor, String beanName) throws BeansException; //-------------------------------------------------------------------------
// Specialized methods for fine-grained control over the bean lifecycle
//------------------------------------------------------------------------- Object createBean(Class beanClass, int autowireMode, boolean dependencyCheck) throws BeansException; Object autowire(Class beanClass, int autowireMode, boolean dependencyCheck) throws BeansException; void autowireBeanProperties(Object existingBean, int autowireMode, boolean dependencyCheck)
throws BeansException; void applyBeanPropertyValues(Object existingBean, String beanName) throws BeansException; Object initializeBean(Object existingBean, String beanName) throws BeansException; Object applyBeanPostProcessorsBeforeInitialization(Object existingBean, String beanName)
throws BeansException; Object applyBeanPostProcessorsAfterInitialization(Object existingBean, String beanName)
throws BeansException; Object resolveDependency(DependencyDescriptor descriptor, String beanName,
Set<String> autowiredBeanNames, TypeConverter typeConverter) throws BeansException; }
这四个接口共同定义了Bena的集合,Bean之间的关系和Bean的行为。
Bean的定义
Bean的定义主要由BeanDefinition描述,类的结构图如下:
Bean的解析
Bean解析的过程非常的复杂,功能被分得很细,因为这里需要被扩展的地方很多,必须保证有足够的灵活性,以应对可能的变化。Bean的解析主要就是对Spring配置文件的解析,这个解析过程如图所示:
以上就包括了Spring IOC中的Bean组件的一些基本的定义和类的结构,请大家参考。
Spring源码分析之IOC容器(一)的更多相关文章
- 【spring源码分析】IOC容器初始化(总结)
前言:在经过前面十二篇文章的分析,对bean的加载流程大致梳理清楚了.因为内容过多,因此需要进行一个小总结. 经过前面十二篇文章的漫长分析,终于将xml配置文件中的bean,转换成我们实际所需要的真正 ...
- 【spring源码分析】IOC容器初始化(二)
前言:在[spring源码分析]IOC容器初始化(一)文末中已经提出loadBeanDefinitions(DefaultListableBeanFactory)的重要性,本文将以此为切入点继续分析. ...
- 【spring源码分析】IOC容器初始化(三)
前言:在[spring源码分析]IOC容器初始化(二)中已经得到了XML配置文件的Document实例,下面分析bean的注册过程. XmlBeanDefinitionReader#registerB ...
- 【spring源码分析】IOC容器初始化(四)
前言:在[spring源码分析]IOC容器初始化(三)中已经分析了BeanDefinition注册之前的一些准备工作,下面将进入BeanDefinition注册的核心流程. //DefaultBean ...
- 【spring源码分析】IOC容器初始化(七)
前言:在[spring源码分析]IOC容器初始化(六)中分析了从单例缓存中加载bean对象,由于篇幅原因其核心函数 FactoryBeanRegistrySupport#getObjectFromFa ...
- 【spring源码分析】IOC容器初始化(十)
前言:前文[spring源码分析]IOC容器初始化(九)中分析了AbstractAutowireCapableBeanFactory#createBeanInstance方法中通过工厂方法创建bean ...
- 【spring源码分析】IOC容器初始化——查漏补缺(一)
前言:在[spring源码分析]IOC容器初始化(十一)中提到了初始化bean的三个步骤: 激活Aware方法. 后置处理器应用(before/after). 激活自定义的init方法. 这里我们就来 ...
- Spring源码分析专题 —— IOC容器启动过程(上篇)
声明 1.建议先阅读<Spring源码分析专题 -- 阅读指引> 2.强烈建议阅读过程中要参照调用过程图,每篇都有其对应的调用过程图 3.写文不易,转载请标明出处 前言 关于 IOC 容器 ...
- 【spring源码分析】IOC容器初始化——查漏补缺(五)
前言:我们知道在Spring中经常使用配置文件的形式对进行属性的赋值,那配置文件的值是怎么赋值到属性上的呢,本文将对其进行分析. 首先了解一个类:PropertySourcesPlaceholderC ...
- 【spring源码分析】IOC容器初始化——查漏补缺(二)
前言:在[spring源码分析]IOC容器初始化(八)中多次提到了前置处理与后置处理,本篇文章针对此问题进行分析.Spring对前置处理或后置处理主要通过BeanPostProcessor进行实现. ...
随机推荐
- JavaScript 正则表达式RegExp 和字符串本身的正则表达式
JavaScript 正则表达式 正则表达式(英语:Regular Expression,在代码中常简写为regex.regexp或RE)使用单个字符串来描述.匹配一系列符合某个句法规则的字符串搜索模 ...
- webpack 项目实践
1,必要的环境 node 环境 下载地址(http://nodejs.cn/) npm 貌似 下载地址(https://www.npmjs.com/)不过,貌似 Node 安装可自带 npm. 2, ...
- 使用crontab调度任务
复杂的.分布式的.工作流式的调度可以通过azkaban来进行调度,除了执行调度任务之外,它还能进行定时调度.而对于简单的服务器任务,如执行一个小脚本,发送邮件等,可以使用crontab命令直接进行,在 ...
- JSP基础知识点
JSP(Java Server Page)是主要有Sun公司倡导的一种动态网页技术,利用JSP可以构建跨平台的动态网站 应用,JSP在服务器端带JSP容器的Web服务器中运行.JSP以Java语言为基 ...
- java-接口和抽象类的联系和区别。
接口和抽象类的联系和区别. 一,简单总结 1.抽象类和接口都不能直接实例化,如果要实例化,抽象类变量必须指向实现所有抽象方法的子类对象,接口变量必须指向实现所有接口方法的类对象. 2.抽象类要被子类继 ...
- springboot项目的重定向和转发
下面是idea软件创建的项目目录,这里总结了一下转发与重定向的问题,详解如下. 首先解释一下每个文件夹的作用,如果你是用的是idea创建的springboot项目,会在项目创建的一开始resource ...
- 如何解决css-子div设置margin-top后,父div与子div一起下移的bug?
根据规范,一个盒子如果没有上补白(padding-top)和上边框(border-top),那么这个盒子的上边距会和其内部文档流中的第一个子元素的上边距重叠. 这是规范引起的普遍问题. 只要给父盒子设 ...
- 配置ftp服务器只能上传不能进行其他操作
又到期末考试了,今年当了数据挖掘助教,课程有一道编程大作业,需要搭建ftp服务器,实现文件上传,但是禁止下载重命名.服务器系统是ubuntu12.04 server,使用的ftp服务器也是linux下 ...
- linux rpm命令之查询包安装与否、包详细信息、包安装位置、文件属于哪个包、包依赖
查询这个包是否已安装rpm -q 包名rpm -qa-q 查询-a all 查询已安装软件包和未安装软件包详细信息 rpm -qi 包名rpm -qip 包全名-i 查询软件信息-p package, ...
- 深入理解net core中的依赖注入、Singleton、Scoped、Transient(一)
相关文章: 深入理解net core中的依赖注入.Singleton.Scoped.Transient(一) 深入理解net core中的依赖注入.Singleton.Scoped.Transient ...