Spring IOC分析
前言
关于Spring,我想无需做太多的解释了。每个Java程序猿应该都使用过他。Spring的ioc和aop极大的方便了我们的开发,但是Spring又有着不好的一面,为了符合开闭原则,Spring的一个方法可以涉及到好几十个类,从设计上来说,这样的设计易于宽展,职责明确。但从开发角度而言,Spring就像一个迷宫,经常会在里面迷失方向。最近,在git上发现tiny-spring。几千行代码就实现了IOC和AOP,看了下很有借鉴意义,对spring的结构清楚了很多。本文开始对IOC做出分析。
对于IOC,我希望从两个问题来讨论
1.什么是IOC容器,它是如何实现的。
2.IOC是如何解决循环依赖的
1.容器的创建与初始化
抛开Spring,假如要求我们自己实现一个容器。应该是如下几步
1.定义一个对象储存bean的信息
public class BeanDefinition {
private Object bean;
private Class beanClass;
private String beanClassName;
}
2.定义一个Map,map的key就是这个bean的名称,value就是bean的BeanDefination
private Map<String, BeanDefinition> beanDefinitionMap = new ConcurrentHashMap<String, BeanDefinition>();
3.每次项目启动的时候把这些bean保存到map里面,需要用到的时候通过bean的name来获得bean的BeanDefination
public Object getBean(String name) {
return beanDefinitionMap.get(name).getBean();
}
ioc容器就是上述这个思路。但是干的事情可比上面那3步复杂多了。
举个例子

XML配置

第一步:读取配置
配置的方式有很多种,xml,注解,代码等都能实现。spring选用的xml作为配置文件,所以第一步就需要读取配置文件。将配置文件的配置信息读到内存中。
下面的代码就是将XML的配置信息填充到Beandefnition。此外还要保存住这些beandefination,所以就需要一个map来保存这些。
private void processProperty(Element ele,BeanDefinition beanDefinition) {
NodeList propertyNode = ele.getElementsByTagName("property");
for (int i = 0; i < propertyNode.getLength(); i++) {
Node node = propertyNode.item(i);
if (node instanceof Element) {
Element propertyEle = (Element) node;
String name = propertyEle.getAttribute("name");
String value = propertyEle.getAttribute("value");
beanDefinition.getPropertyValues().addPropertyValue(new PropertyValue(name,value));
}
}
}
第二步:生产bean
配置信息都已经读取到了,接下来就可以开始创建bean了。创建bean这个工作我们交给BeanFactory来完成。这个时候的创建bean总共可以分作两步
1.根据类的全限定名反射生成javaBean
2.将从xml中的读取道德配置信息设置到javaBean中
protected Object doCreateBean(BeanDefinition beanDefinition) throws Exception {
Object bean = createBeanInstance(beanDefinition);
applyPropertyValues(bean, beanDefinition);
return bean;
}
protected Object createBeanInstance(BeanDefinition beanDefinition) throws Exception {
return beanDefinition.getBeanClass().newInstance();
}
protected void applyPropertyValues(Object bean, BeanDefinition mbd) throws Exception {
for (PropertyValue propertyValue : mbd.getPropertyValues().getPropertyValues()) {
Field declaredField = bean.getClass().getDeclaredField(propertyValue.getName());
declaredField.setAccessible(true);
declaredField.set(bean, propertyValue.getValue());
}
}
第三步:
这一步其实就是从map中获取bean,没什么好多的,但值得注意的是,但spring设置成懒加载的时候,这一步就会像第二步一样去创建javaBean。
2.循环依赖的解决
看以下代码
public class A {
private B b;
public B getB() {
return b;
}
public void setB(B b) {
this.b = b;
}
}
public class B {
private A a;
public A getA() {
return a;
}
public void setA(A a) {
this.a = a;
}
}
两个类互相持有对方的引用,A在设置自己属性的时候发现B还没有被实例化,这时候就去要去实例化B,然后b在设置自己属性的时候发现A还没有实例化完成,就去实例化A,这就会导致循环依赖问题。
spring新增BeanReference对象,来表示这个属性是对另一个bean的引用。这个在读取xml的时候初始化,并在初始化bean的时候,进行解析和真实bean的注入。
举个例子,下面两个bean互相持有对方引用

读取xml的时候初始化
String ref = propertyEle.getAttribute("ref");
if (ref == null || ref.length() == 0) {
throw new IllegalArgumentException("Configuration problem: <property> element for property '"
+ name + "' must specify a ref or value");
}
BeanReference beanReference = new BeanReference(ref);
beanDefinition.getPropertyValues().addPropertyValue(new PropertyValue(name, beanReference));
解析获取bean
protected void applyPropertyValues(Object bean, BeanDefinition mbd) throws Exception {
for (PropertyValue propertyValue : mbd.getPropertyValues().getPropertyValues()) {
Field declaredField = bean.getClass().getDeclaredField(propertyValue.getName());
declaredField.setAccessible(true);
Object value = propertyValue.getValue();
if (value instanceof BeanReference) {
BeanReference beanReference = (BeanReference) value;
value = getBean(beanReference.getName());
}
declaredField.set(bean, value);
}
}
我们再看下getBean的方法


在注入bean的时候,如果bean不存在,则创建bean。创建完成之后,再去注入。这样就解决了循环依赖问题。
3.IOC流程简述
以上通过两个问题去探索IOC,这样的介绍会有点不够体系化。接下来跟着代码从头到尾运行一遍。
测试代码如下

ClassPathXmlApplicationContext继承ApplicationContext,而ApplicationContext是BeanFactory的子集。相比而言,ApplicationContext对bean的一些操作做了封装 使用起来更加方便 下面的代码是不用ApplicationContext时我们获取bean需要的步骤。对于开发者而言,ApplicationContext无疑更加友好

现在点进ClassPathXmlApplicationContext


这里主要看refresh

现在
已经执行完毕了。容器的大致框架已经建立,但这个时候bean还没生成。我们接着看下一行。
getBean的方法点进去

真正获取bean操作的还是beanFactory,这也说明了ApplicationContext是对beanFactory的封装

再看doCreateBean

设置属性

至此
这一行代码已经结束了,容器里面的bean都生成好了。bean也取出来了,下一步执行方法即可。
总结
本文通过两个问题描述了ioc容器的基本情况,最后又总体介绍了大致的流程。这只是最精简版的ioc容器初始化bean的过程,下一步会把aop也加进来。
Spring IOC分析的更多相关文章
- spring IOC 分析及实现
什么是IOC Inversion of Control,控制反转,也成依赖倒置. 反转: 依赖对象的创建被反转,使用IOC之前,对象由自己创建,反转后,由IOC容器获取 IOC容器的工作: 负责创建, ...
- 框架源码系列二:手写Spring-IOC和Spring-DI(IOC分析、IOC设计实现、DI分析、DI实现)
一.IOC分析 1. IOC是什么? IOC:Inversion of Control控制反转,也称依赖倒置(反转) 问题:如何理解控制反转? 反转:依赖对象的获得被反转了.由自己创建,反转为从IOC ...
- spring IOC源码分析(1)
1.何谓Spring IOC 何谓Spring IOC?书上谓之“依赖注入”,那何谓“依赖注入”? 作为一个Java程序猿,应该遇到过这样的问题,当你在代码中需要使用某个类提供的功能时,你首先需要ne ...
- Spring IOC容器分析(2) -- BeanDefinition
上文对Spring IOC容器的核心BeanFactory接口分析发现:在默认Bean工厂DefaultListableBeanFactory中对象不是以Object形成存储,而是以BeanDefin ...
- Spring IOC容器分析(3) -- DefaultListableBeanFactory
上一节介绍了封装bean对象的BeanDefinition接口.从前面小结对BeanFactory的介绍中,我们知道bean对象是存储在map中,通过调用getBean方法可以得到bean对象.在接口 ...
- Spring IOC容器分析(4) -- bean创建获取完整流程
上节探讨了Spring IOC容器中getBean方法,下面我们将自行编写测试用例,深入跟踪分析bean对象创建过程. 测试环境创建 测试示例代码如下: package org.springframe ...
- Spring的IOC分析(一)
我们学习Spring之前需要对23种java的设计模式的9种有一定的理解,设计模式为了解耦,Spring也是在解耦的方向上设计的,所以设计模式要理解一下,它当中用到了很多. 单例模式(写法很多钟,7种 ...
- Spring源码分析:Spring IOC容器初始化
概述: Spring 对于Java 开发来说,以及算得上非常基础并且核心的框架了,在有一定开发经验后,阅读源码能更好的提高我们的编码能力并且让我们对其更加理解.俗话说知己知彼,百战不殆.当你对Spri ...
- Spring之旅第二篇-Spring IOC概念及原理分析
一.IOC概念 上一篇已经了解了spring的相关概念,并且创建了一个Spring项目.spring中有最重要的两个概念:IOC和AOP,我们先从IOC入手. IOC全称Inversion of Co ...
随机推荐
- 第五周助教工作总结——NWNU李泓毅
第五周助教总结 注:因第四次实验安排两个标准时间完成,因此本周未提交完整作业. 本周心得: 第四次实验进行过半,八组同学都在实验课上进行了一次中期总结,并形成书面总结在微信群中讨论. 根据各组同学的中 ...
- web安全系列4:google语法
这是web安全的第四篇,欢迎翻看前面几篇. 前面我们介绍了一些和HTTP有关知识,那么一个疑问就是黑客要做的第一件是什么?其实很简单,确定一个目标,然后搜集信息. 这很容易理解,我们无论做什么都得先有 ...
- go语言的条件语句和循环语句
一,条件语句 常见的就是if语句: 单支条件语句: if 条件 :执行语句 (注,如果是没有逻辑运算符连接的话,是可以不需要括号的,也可以加上括号,如:if (条件):执行语句) 双支 ...
- Python的条件判断语句------if/else语句
计算机之所以能做很多自动化的任务,因为它可以自己做条件判断. 比如,输入用户的年龄,根据年龄打印不同的内容... Python程序中,能让计算机自己作出判断的语句就是if语句: 例: age = 25 ...
- shell awk处理过滤100万条数据
背景: 100万条数据.格式如下: ID 地址 1895756546931805 安徽省六安市裕安区固镇镇佛俺村柳树队5758 安徽省蒙城县岳坊镇胡寨村小组小胡寨庄6号 183494167409969 ...
- shell搭建CentOS_7基础环境
#!/bin/bash#Auth:Darius#CentOS_7配置实验环境eno=`ifconfig|awk '{print $1}'|head -1|awk -F ":" '{ ...
- python_flask框架学习之路(1)
1.初识web,了解utl . 术语: scheme://host:port/path?query-string=xxx#yyyy 例子:https://i.cnblogs.com/EditArtic ...
- can't open the mysql.plugin table. please run mysql_upgrade to create it.
To initialize a fresh data directory, you basically (after setting your config file) just have to ru ...
- 为程序设置多语言界面——C#
考虑到程序的国际化需求,需要为程序设置多语言界面. 1,新建一个资源文件,名字可以是对应界面+语言代码(MainForm.zh-CN).这样资源文件就会自动添加到对应界面下面. 2,更改界面属性Loc ...
- c++实现对windwos 下socket 的封装(实现封包及拆包处理)
SuperSocket.h #pragma once #include<string> #include<iostream> #include <WINSOCK2.H&g ...