Spring IoC源码解读——谈谈bean的几种状态
阅读Spring IoC部分源码有一段时间了,经过不断的单步调试和参阅资料,对Spring容器中bean管理有了一定的了解。这里从bean的几个状态的角度出发,研究下IoC容器。
一、原材料
Xml中的bean定义配置(或者注解)、及Java代码
<bean id="book" name="book" class="com.sky.vo.Book" scope="singleton" init-method="productBook" destroy-method="destroyBook">
<property name="title" value="小王子"/>
</bean>
public class Book implements InitializingBean, DisposableBean, BeanFactoryAware, ApplicationContextAware{
private String title;
public Book(){
System.out.println("调用了Book默认构造函数");
}
public void productBook(){
System.out.println("书本初始化init-method");
}
///.......
}
二、半成品
BeanDefinition是Spring中很重要的一个接口,作用是对bean的一种抽象(简单的理解是他的属性对应了<bean>中定义的属性和子节点等信息)。他是半成品,所以我们几乎不会直接使用,而是在IoC容器内部流通!
阅读源码发现是用SAX解析xml,然后保存到BeanDefinition对象。以下是关键的伪代码
/*
**BeanDefinitionParserDelegate.java
*/
public AbstractBeanDefinition parseBeanDefinitionElement(Element ele, String beanName, BeanDefinition containingBean) {
AbstractBeanDefinition bd = createBeanDefinition(className, parent){
GenericBeanDefinition bd = new GenericBeanDefinition();
bd.setBeanClassName(className);
return bd;
}
parseBeanDefinitionAttributes(ele, beanName, containingBean, bd);
parseConstructorArgElements(ele, bd);
parsePropertyElements(ele, bd);
//省略了解析其他子节点的代码
}
核心的解析过程比较简单,创建了GenericBeanDefinition,然后节点属性和<property>、<construct-arg>等设置为他的属性。
那么,还有一个比较关键的步奏,就是转化后的BeanDefinition存放的时机和位置,是在Spring IoC容器启动时存放的,具体代码:
/*
**DefaultListableBeanFactory.java
*/
public void registerBeanDefinition(String beanName, BeanDefinition beanDefinition){
synchronized (this.beanDefinitionMap) {
Object oldBeanDefinition = this.beanDefinitionMap.get(beanName);
if (oldBeanDefinition != null) {
if (!this.allowBeanDefinitionOverriding) {
throw new BeanDefinitionStoreException("Cannot register bean definition for bean , There is already bound.");
}
else {
this.logger.info("Overriding bean definition for bean, replacing oldBeanDefinition");
}
}
else {
this.beanDefinitionNames.add(beanName);
}
this.beanDefinitionMap.put(beanName, beanDefinition);
}
}
可以看到最终存储到了beanDefinitionMap这个Map中,来看看他的定义(DefaultListableBeanFactory.java中)
private final Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<String, BeanDefinition>(64);
这样,Spring容器就持有了BeanDefinition。
三、成品
获取成品就是获取具体的bean对象,其实就是Object bean = BeanFactroy.getBean(beanName).这里根据scope的不同,会有一些不同,简而言之:
A、singleton在IoC容器初始化时实例化,并缓存到Map中;getBean(singletonBeanName)时从这个Map中取即可;
如果singleton同时lazy-init="true",第一次getBean时缓存到Map,以后从Map中取;
B、getBean(prototypeBeanName)时每次都实例化新的bean对象;
1、singleton的bean
直接看伪代码(截取了部分关键代码)
//从getBean入手
BeanFactroy.getBean("beanName"){
AbstractBeanFactory.createBean{
Object beanInstance = doCreateBean(beanName, mbd, args){
// Eagerly check singleton cache for manually registered singletons.
//这是返回了缓存的singleton的bean:com.sky.vo.Book@1a1ecd
Object sharedInstance = getSingleton(beanName){
return Object singletonObject = this.singletonObjects.get(beanName);
}
return (T)bean;
}
}
}
从源码我们可以看到,对于singleton的bean,我们直接从singletonObjects这个Map中取!
public class DefaultSingletonBeanRegistry{
/**缓存了singleton的bean:beanName--->bean实例*/
private final Map<String, Object> singletonObjects = new ConcurrentHashMap<String, Object>(64);
}
那么bean实例是什么时候存入到这个map的呢?看源码知道是在IoC容器启动时。
通过多次分析源码,我们知道AbstractApplicationContext.refresh()完成了容器的初始化过程,singleton的bean的实例化也是在这完成的。
public void AbstractApplicationContext.refresh(){
//创建Spring容器
ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();
// Instantiate all remaining (non-lazy-init) singletons.
AbstractApplicationContext.finishBeanFactoryInitialization(beanFactory){
DefaultListableBeanFactory.preInstantiateSingletons(){
RootBeanDefinition bd = getMergedLocalBeanDefinition(beanName);
if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {
//然后,执行了getBean的操作!!!这里执行的代码就是BeanFactory.getBean(beanName)部分
AbstractBeanFactory.getBean(beanName){
AbstractBeanFactory.doGetBean(beanName){
//缓存为空
Object sharedInstance = getSingleton(beanName);
//如果是singleton
if (mbd.isSingleton()) {
DefaultSingletonBeanRegistry.getSingleton(){
synchronized (this.singletonObjects) {
//这里缓存也是空的
Object singletonObject = this.singletonObjects.get(beanName);
//实例化bean
singletonObject = singletonFactory.getObject(){
Object beanInstance = doCreateBean(beanName, mbd, args);
}
//这里非常重要,把实例化的bean放到map中
addSingleton(beanName, singletonObject){
this.singletonObjects.put(beanName, (singletonObject != null ? singletonObject : NULL_OBJECT));
}
}
}
}
}
}
}
}
}
}
截取了部分关键的代码片段,可以看到Spring容器初始化时会把实例化的singleton的bean放到singletonObject
Spring IoC源码解读——谈谈bean的几种状态的更多相关文章
- Spring Ioc源码分析系列--Bean实例化过程(一)
Spring Ioc源码分析系列--Bean实例化过程(一) 前言 上一篇文章Spring Ioc源码分析系列--Ioc容器注册BeanPostProcessor后置处理器以及事件消息处理已经完成了对 ...
- Spring Ioc源码分析系列--Bean实例化过程(二)
Spring Ioc源码分析系列--Bean实例化过程(二) 前言 上篇文章Spring Ioc源码分析系列--Bean实例化过程(一)简单分析了getBean()方法,还记得分析了什么吗?不记得了才 ...
- spring beans源码解读之--Bean的注解(annotation)
随着spring注解的引入,越来越多的开发者开始使用注解,这篇文章将对注解的机制进行串联式的讲解,不求深入透彻,但求串起spring beans注解的珍珠,展示给大家. 1. spring beans ...
- spring beans源码解读之--Bean的定义及包装
bean的定义,包装是java bean的基础.再怎么强调它的重要性都不为过,因此深入 了解这块的代码对以后的代码研究可以起到事半功倍的功效. 1. Bean的定义BeanDefinition 1.1 ...
- spring beans源码解读之--bean definiton解析器
spring提供了有两种方式的bean definition解析器:PropertiesBeanDefinitionReader和XmLBeanDefinitionReader即属性文件格式的bean ...
- Spring Ioc源码分析系列--容器实例化Bean的四种方法
Spring Ioc源码分析系列--实例化Bean的几种方法 前言 前面的文章Spring Ioc源码分析系列--Bean实例化过程(二)在讲解到bean真正通过那些方式实例化出来的时候,并没有继续分 ...
- Spring Ioc源码分析系列--@Autowired注解的实现原理
Spring Ioc源码分析系列--@Autowired注解的实现原理 前言 前面系列文章分析了一把Spring Ioc的源码,是不是云里雾里,感觉并没有跟实际开发搭上半毛钱关系?看了一遍下来,对我的 ...
- Spring Ioc源码分析系列--自动注入循环依赖的处理
Spring Ioc源码分析系列--自动注入循环依赖的处理 前言 前面的文章Spring Ioc源码分析系列--Bean实例化过程(二)在讲解到Spring创建bean出现循环依赖的时候并没有深入去分 ...
- 深入Spring IOC源码之ResourceLoader
在<深入Spring IOC源码之Resource>中已经详细介绍了Spring中Resource的抽象,Resource接口有很多实现类,我们当然可以使用各自的构造函数创建符合需求的Re ...
随机推荐
- 解决h5的video标签,android、ipad客户端播放正常,iphone客户端无法播放
在做html5时插入一个视频播放标签video后,测试时android.ipad客户端播放正常,唯独iphone自带浏览器无法播放. 解决办法: 判断用户所使用客户端访问h5页面时是iphone时,点 ...
- dpi 、 dip 、分辨率、屏幕尺寸、px、density 关系以及换算
一.基本概念 dip : Density independent pixels ,设备无关像素. dp :就是dip px : 像素 dpi :d ...
- Unity 3D制作2D游戏的几种方法
1.使用本身UGUI. 2.把摄像机的投影改为正交投影,不考虑Z轴. 3.使用Untiy自身的2D模式. 4.使用2D TooKit插件.
- JavaScript面向对象程序设计:数组
或许你会奇怪,面向对象的程序设计为什么从数组开始讲起?这是因为……其间的种种关系吧……嘿嘿,这里先卖个关子,先来看看我们熟悉的数组在JavaScript里面是什么样子的. 1. 创建数组 在J ...
- Linux - 日志文件
Linux日志文件绝大多数存放在/var/log目录,其中一些日志文件由应用程序创建,其他的则通过syslog来创建. Linux系统日志文件通过syslog守护程序在syslog套接字/dev/lo ...
- ionic+nodejs开发遇到的跨域和post请求数据问题
最近学习ionic+nodejs开发混合app中遇到了一些问题,在此总结一下. 开发环境搭建 项目地址 https://github.com/ytudt/nodejsApp 代码和问题都会在之后的学习 ...
- 如何调试PHP程序
一.PHP自带的调试功能 1.修改php.ini )开发环境 需要打开报错输出显示,方便开发者调试. display_errors = On )生产环境 不能直接将错误输出,而是记入日志,以免透露路径 ...
- Mina架构与优化指南
MINA架构 这里,我借用了一张Trustin Lee在Asia 2006的ppt里面的图片来介绍MINA的架构. Remote Peer就是客户端,而下方的框是MINA的主要结构,各个框之间的箭头代 ...
- Windows Azure Cloud Service (38) 微软IaaS与PaaS比较
<Windows Azure Platform 系列文章目录> 最近一直想总结Azure IaaS和PaaS的区别与比较,写个博文详细说明一下.建议读者在阅读之前,先熟悉微软PaaS和Ia ...
- Azure REST API (2) Azure Storage
<Windows Azure Platform 系列文章目录> 注意:本文适用于国内由世纪互联运维的Azure China. 本文将会介绍如何使用REST API来直接访问Storage ...