Spring IoC的一些知识点
在日常开发中,接触得比较多的算是Spring生态了,Spring Ioc是Spring Framework重要的组成部分,下面整理了一些Spring Ioc的知识点。
1. 什么是IoC
IoC(Inversion of Control),翻译过来就是控制反转,它是一种设计思想,这个设计思想说明了 “由谁控制,控制了什么,为何反转,反转了什么”。
- 控制
Java通过new来创建对象,而IoC有专门的容器控制对象创建,即Ioc控制了对象的创建(实际还包含更多的外部资源,如配置等)
- 反转
使用IoC容器管理对象时,对象持有的依赖对象是由IoC注入的,即对象只是被动的接受了依赖对象,也即依赖对象的获取反转了。
如:需要对 UserService 这个类进行单元测试,其中 UserService 还依赖 UserAddressService:
public class UserServiceTests {
public void test() {
// ...
}
}
public class UserService {
public UserAddressService userAddressService;
}
public class UserAddressService {
}
那么在没有IoC的情况下,UserServiceTests.test() 测试流程为:
- 创建UserService对象
- 创建UserAddressService对象
- 将UserAddressService对象注入到UserService中
而在使用IoC的情况下,UserServiceTests.test() 测试流程则为:
- UserServiceTests获取UserService对象
- IoC控制:
- 创建UserService对象
- 创建UserAddressService对象
- IoC反转:
- 将UserAddressService对象注入到UserService中
可以用“别找我们,我们找你”来总结IoC的思想,即由IoC容器帮对象找相应的依赖对象注入,而不是由对象主动去找依赖对象注入。
2. IoC的另一个角度
DI(Dependency Injection),翻译过来就是依赖注入,可以理解为从另一个角度的IoC思想,这里的依赖注入是指:
- 依赖
应用程序依赖了IoC容器,应用程序需要IoC来提供对象需要的资源(包括依赖对象,配置等)
- 注入
IoC容器注入应用程序需要的某个对象以及应用程序依赖的对象(实际还包含更多的外部资源,如配置等)。
3. BeanFactory和ApplicationContext
Spring的 org.springframework.beans
和 org.springframework.context
包是IoC的基础,其中:
- BeanFactory 接口提供了IoC容器最基本功能
- ApplicationContext 是BeanFactory的子接口,提供了更多的功能,包含SpringAOP,i18n国际化,事件,以及不同层次的context实现(如WebApplicationContext)
下面举例两者构建IoC容器的方式,这里使用XML直接开启注解的方式声明Bean:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
https://www.springframework.org/schema/context/spring-context.xsd">
<context:component-scan base-package="io.michong2022.spring.bean.demo"/>
</beans>
上面spring.xml的 <context:component-scan>
配置会启用 <context:annotation-config/>
,而<context:annotation-config/>
会注册下面的Bean:
- ConfigurationClassPostProcessor
- AutowiredAnnotationBeanPostProcessor
- CommonAnnotationBeanPostProcessor
- PersistenceAnnotationBeanPostProcessor
- EventListenerMethodProcessor
在 io.michong2022.spring.bean.demo
包中声明了两个Bean(UserService, UserAddressService):
/**
* @author 米虫2022
*/
@Service
public class UserService {
@Autowired
private UserAddressService userAddressService;
}
BeanFactory构建IoC容器:
@Test
public void testBeanFactory() {
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();
XmlBeanDefinitionReader reader = new XmlBeanDefinitionReader(beanFactory);
reader.loadBeanDefinitions("spring.xml");
beanFactory.addBeanPostProcessor(beanFactory.getBean(AutowiredAnnotationBeanPostProcessor.class));
UserService userService = beanFactory.getBean(UserService.class);
Assertions.assertNotNull(userService);
}
ApplicationContext构建IoC容器:
@Test
public void testApplicationContext() {
ApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
UserService userService = context.getBean(UserService.class);
Assertions.assertNotNull(userService);
}
这里会加载7个Bean,2个自定义的,以及5个context:annotation-config注册的:
Loaded 7 bean definitions from class path resource [spring.xml]
默认情况下自定义的Bean是单例的:
Creating shared instance of singleton bean 'userService'
4. IoC Bean的构建
针对Ioc Bean的构建,BeanFactory和ApplicationContext的行为有些许不同,以ClassPathXmlApplicationContext为例,它会在refresh()
的时候,
通过 finishBeanFactoryInitialization(beanFactory)
将Bean都初始化,而BeanFactory则在getBean()
的时候才真正的创建Bean。
以单例Bean的构建为例,Bean构建流程如下:
其中:Bean单例对象由DefaultSingletonBeanRegistry的singletonObjects管理,DefaultListableBeanFactory的父类继承了DefaultSingletonBeanRegistry。
5. Bean的循环依赖
Bean的循环依赖,指Bean直接或间接的依赖构成了环,如:A -> B, B -> C, C -> A
,下面是一个例子:
@Service
public class UserService {
// UserService依赖UserAddressService
@Autowired
private UserAddressService userAddressService;
}
@Service
public class UserAddressService {
// UserAddressService也依赖UserService
@Autowired
private UserService userService;
}
- Spring解决了单例Bean的Setter的循环依赖,对于构造函数循环依赖和非单例的Bean循环依赖会抛出异常。
DefaultSingletonBeanRegistry中有一些容器管理着Bean创建过程的状态:
- singletonObjects: 创建成功的单例Bean对象缓存池
- singletonFactories: 提前暴露的Bean对象工厂缓存池,工厂提供Bean完成了实例化,并没有完成依赖注入
- earlySingletonObjects: 提前暴露的Bean对象缓存池
- registeredSingletons: 已经注册的Bean名称
- singletonsCurrentlyInCreation: 正在创建的Bean
Bean的依赖注入流程如下:
在流程图的黄色部分,Spring会根据下面的顺序获取依赖的Bean对象:
- 尝试从
singletonObjects
获取已经成功创建的Bean - 如果(1)获取不到,判断需要获取的Bean是否处理构建中,即是否登记在
singletonsCurrentlyInCreation
中 - 如果需要获取的Bean登记在
singletonsCurrentlyInCreation
中,那么尝试从earlySingletonObjects
获取Bean - 如果(3)获取不到,那么从singletonFactories获取Bean的工厂,获取Bean对象,将Bean对象放入
earlySingletonObjects
中,且将工厂从singletonFactories
移除
注意:从Bean工厂获取的Bean,并不是完成注入的Bean对象。
按上面的流程,例子中,测试用例 beanFactory.getBean(UserService.class);
获取Bean执行的流程如下:
- 构建UserService,登记
singletonsCurrentlyInCreation
和singletonFactories
- 为UserService注入UserAddressService对象(UserAddressService此时没有创建,暂停执行步骤3.)
- 构建UserAddressService,登记
singletonsCurrentlyInCreation
和singletonFactories
- 为UserAddressService注入UserService对象,从singletonFactories获取UserService工厂,从而获取UserService对象(UserAddressService属性为空)放入earlySingletonObjects中,并将UserService的工厂从singletonFactories移除,将获取的UserService对象注入
- UserAddressService对象构建完成,将放入singletonObjects中,并将UserAddressService工厂从singletonFactories移除
- 完成Bean的构建流程,在
afterSingletonCreation
中从singletonsCurrentlyInCreation移除 - 继续步骤2.
下面是流程图黄色部分的源码实现:
@Nullable
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
// Quick check for existing instance without full singleton lock
Object singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null && allowEarlyReference) {
synchronized (this.singletonObjects) {
// Consistent creation of early reference within full singleton lock
singletonObject = this.singletonObjects.get(beanName);
if (singletonObject == null) {
singletonObject = this.earlySingletonObjects.get(beanName);
if (singletonObject == null) {
ObjectFactory<?> singletonFactory = this.singletonFactories.get(beanName);
if (singletonFactory != null) {
singletonObject = singletonFactory.getObject();
this.earlySingletonObjects.put(beanName, singletonObject);
this.singletonFactories.remove(beanName);
}
}
}
}
}
}
return singletonObject;
}
不支持循环依赖的Bean构建:
- 构造函数注入循环依赖,抛
org.springframework.beans.factory.UnsatisfiedDependencyException
异常 - 非单例对象注入循环依赖,抛
org.springframework.beans.factory.UnsatisfiedDependencyException
异常
7. 自动装配
自动装配是指Spring自动的注入对象,Spring可以通过注解@Autowired
来完成自动装配,默认情况下,@Autowired优先通过属性名称注入,即byName
注入优先。
下面这个例子,自动装配会发生异常:
public interface Address {
}
@Service("provinceAddress")
public class ProvinceAddress implements Address {
}
@Service("cityAddress")
public class CityAddress implements Address {
}
@Service
public class UserService {
@Autowired
private Address address;
}
出现如下异常:
No qualifying bean of type 'io.michong2022.spring.bean.demo.Address' available: expected single matching bean but found 2: cityAddress,provinceAddress
可以通过如下方式解决:
- 修改变量名
address
为想要注入的Bean名称,如cityAddress或provinceAddress - 为CityAddress或ProvinceAddress加上
@Primary
注解,标注为默认的注入对象 - 给变量
address
加上@Qualifier("cityAddress")
注解,指定注入的对象
8. Bean的生命周期
Spring Bean的声明周期如下:
- 构造函数
- BeanPostProcessor.postProcessBeforeInitialization()
- @PostConstruct (CommonAnnotationBeanPostProcessor)
- afterPropertiesSet (InitializingBean)
- init-method (Bean配置指定)
- BeanPostProcessor.postProcessAfterInitialization()
- @PreDestroy (CommonAnnotationBeanPostProcessor)
- destroy (DisposableBean)
- destroy-method (Bean配置指定)
Spring IoC的一些知识点的更多相关文章
- Spring IOC知识点一网打尽!
前言 只有光头才能变强 回顾前面: 给女朋友讲解什么是代理模式 包装模式就是这么简单啦 单例模式你会几种写法? 工厂模式理解了没有? 在刷Spring书籍的时候花了点时间去学习了单例模式和工厂模式,总 ...
- spring中的IOC/DI的知识点
IOC(Inverse of control):控制反转;其实就是一个装对象的容器,以前我们在controller中调用service中的方法,都要先new 一个service对象,这样不符合模式设计 ...
- spring原理案例-基本项目搭建 03 创建工程运行测试 spring ioc原理实例示例
下面开始项目的搭建 使用 Java EE - Eclipse 新建一 Dynamic Web Project Target Runtime 选 Apache Tomcat 7.0(不要选 Apache ...
- 案例学编程系列:案例认识 Spring IOC
本文spring libs 地址:https://github.com/yizhiamumu/springlibs Spring 能帮我们做什么 ①.Spring 能帮我们根据配置文件创建及组装对象之 ...
- Spring IOC 容器源码分析
声明!非原创,本文出处 Spring 最重要的概念是 IOC 和 AOP,本篇文章其实就是要带领大家来分析下 Spring 的 IOC 容器.既然大家平时都要用到 Spring,怎么可以不好好了解 S ...
- 关于Spring IOC (DI-依赖注入)需要知道的一切
关联文章: 关于Spring IOC (DI-依赖注入)你需要知道的一切 关于 Spring AOP (AspectJ) 你该知晓的一切 <Spring入门经典>这本书无论对于初学者或者有 ...
- Spring IOC 源码分析
Spring 最重要的概念是 IOC 和 AOP,本篇文章其实就是要带领大家来分析下 Spring 的 IOC 容器.既然大家平时都要用到 Spring,怎么可以不好好了解 Spring 呢?阅读本文 ...
- 关于Spring IOC (DI-依赖注入)
<Spring入门经典>这本书无论对于初学者或者有经验的工程师还是很值一看的,最近花了点时间回顾了Spring的内容,在此顺带记录一下,本篇主要与spring IOC相关 ,这篇博文适合初 ...
- Spring IOC 巨多 非常 有用
关联文章: 关于Spring IOC (DI-依赖注入)你需要知道的一切 关于 Spring AOP (AspectJ) 你该知晓的一切 <Spring入门经典>这本书无论对于初学者或者有 ...
- Spring IOC 容器源码分析(转)
原文地址 Spring 最重要的概念是 IOC 和 AOP,本篇文章其实就是要带领大家来分析下 Spring 的 IOC 容器.既然大家平时都要用到 Spring,怎么可以不好好了解 Spring 呢 ...
随机推荐
- 从零开始学Graph Database:什么是图
摘要:本文从零开始引导与大家一起学习图知识.希望大家可以通过本教程学习如何使用图数据库与图计算引擎.本篇将以华为云图引擎服务来辅助大家学习如何使用图数据库与图计算引擎. 本文分享自华为云社区<从 ...
- 删除数组里含有a的元素,并且将null值放在后面
想去掉里面含有a的元素,并将null放在后面.放在后面就是往后移,其他值往左移 1 public static void main(String[] args) { 2 //自定义的一个数组 3 St ...
- MasaFramework -- 缓存入门与设计
概念 什么是缓存,在项目中,为了提高数据的读取速度,我们会对不经常变更但访问频繁的数据做缓存处理,我们常用的缓存有: 本地缓存 内存缓存:IMemoryCache 分布式缓存 Redis: Stack ...
- go-zero docker-compose 搭建课件服务(二):编写courseware rpc服务
0.转载 go-zero docker-compose 搭建课件服务(二):编写courseware rpc服务 0.1源码地址 https://github.com/liuyuede123/go-z ...
- python不确定性计算之粗糙集属性约简
粗糙集属性约简 本实验同时采用区别矩阵和依赖度约简. 在依赖度约简中,设置依赖度计算函数和相对约简函数,对读取的数据进行处理,最后根据依赖度约简. 在读取数据后判断有无矛盾,若有则进行决策表分解,然后 ...
- GIT入门与Gitee的使用
一:Git是什么? Git是目前世界上最先进的分布式版本控制系统. 工作原理 / 流程: Workspace:工作区Index / Stage:暂存区Repository:仓库区(或本地仓库)Remo ...
- nginx+keepalived实现主从模式双机热备份
主从模式就是一台机器提供服务,另一台机器作为备份机,当主机的服务停止时,备份机立刻接替主机的服务. 安装 安装nginx wget http://nginx.org/download/nginx-1. ...
- CentOS6/7 配置守护进程
CentOS6.x CentOS6中转用Upstrat代替以前的init.d/rcX.d的线性启动方式. 一.相关命令 通过initctl help可以查看相关命令 [root@localhost ~ ...
- javascript异步编程之generator(生成器函数)与asnyc/await语法糖
Generator 异步方案 相比于传统回调函数的方式处理异步调用,Promise最大的优势就是可以链式调用解决回调嵌套的问题.但是这样写依然会有大量的回调函数,虽然他们之间没有嵌套,但是还是没有达到 ...
- 2022春每日一题:Day 41
题目:I Hate It 一个基础的线段树模板,单点修改+区间查询 代码: #include <cstdio> #include <cstdlib> #include < ...