Spring 是如何造出一个 Bean 的
前言
使用 Java 作为第一开发语言的朋友们,相信大家或多或少的都使用过 Spring 这个开发框架,可以说 Spring 框架真是我们 Java 程序员的春天,在 Spring 中 Bean 是其中最重要的概念之一,是学习其它高级知识的基础,Bean 说白了其实就是一个被 Spring 框架管理的对象,今天我们来看看 Bean 在 Spring 中是如何被造出来的。
1. Bean 要如何定义
假如你有如下这样的一个 Programmer 类,这个程序员类有三个属性: 姓名(name)、年龄(age)、是否有女朋友(hasGirlFriend)(P.S. 正常情况下 hasGirlFriend 属性应该都是 false),还有一个显示个人资料的方法 showMaterial。
/**
* @Author: mghio
* @Date: 2020-10-05
* @Description: Programmer.
*/
public class Programmer {
private String name;
private Integer age;
private Boolean hasGirlFriend;
public void showMaterial() {
System.out.println("name: " + name + ", age: " + age + ", hasGirlFriend: " + hasGirlFriend);
}
}
现在请你思考一下,如果让你来设计该如何在一个 Spring 容器中描述这样的一个 Programmer 对象呢?
无非就是需要如下这些信息:
1.1 类名
首先类名肯定是需要的,这样到时候才能通过类名加载到这个类。
1.2 实例别名
当我们在一个容器中如果一个类有多个实例或者不想通过一个类名来描述一个实例时,这时通过设置一个别名就可以很方便的描述该实例了。
1.3 构造函数
我们知道 Java 中创建一个类的实例首先就会调用该类的构造函数,当有多个构造函数时,需要明确的描述要使用哪个构造函数来创建对象,比如通过传入不同的参数类型来选择不同的构造函数。
1.4 类的属性设置
当我们没有在构造函数中传入属性,比如上面的 Programmer 可以直接通过无参构造函数就可以创建出来了,后面如果需要设置实例的属性则需要调用其设置属性的方式来进行设置,所以属性方法也是必要的。
1.5 初始化方法
有时候我们需要在一个实例化完成之后做一些我们自定义的业务逻辑,比如想让上面例子中的 Programmer 在实例化完成之后就显示个人资料(调用 showMaterial() 方法),这种场景使用初始化方法就很合适了。
1.6 销毁方法
说到销毁,大家可能都会想到和资源有关,比如一个共识就是大家一般都把资源释放类的工作放在 finally 代码块中确保资源可以得到释放,同样当一个 Bean 之后连接使用了某些资源时,当销毁后想要对这些资源进行释放,这时候就可以通过其 销毁方法 来释放资源。
1.7 作用域
有些 Bean 可能需要在整个容器中只有一个,也就是单例,而有些可能要求每一次请求对应的 Bean 都不一样,这时可以通过一个 作用域 的概念,来区分不同要求的 Bean,当容器发现这个类是 单例 的,就会复用已存在的 Bean,否则才重新创建。
当然这里只是列举一些个人觉得比较重要的属性,还有其它的一些属性需要增加。在 Spring 框架中 Bean 的定义是通过一个 BeanDefinition 类来描述的。
在没有使用 SpringBoot 之前我们都是通过 XML 配置然后 Spring 来解析生成 Bean 的,同时我们也可以通过代码方式使用 BeanDefinitionBuilder 生成 Bean,具体代码如下所示:
/**
* @Author: mghio
* @Date: 2020-10-05
* @Description:
*/
public class ProgrammerTest {
public static void main(String[] args) {
new Programmer().showMaterial();
BeanDefinitionBuilder beanDefinitionBuilder = BeanDefinitionBuilder.genericBeanDefinition(Programmer.class);
beanDefinitionBuilder.addPropertyValue("name", "mghio");
beanDefinitionBuilder.addPropertyValue("age", 18);
beanDefinitionBuilder.addPropertyValue("hasGirlFriend", false);
DefaultListableBeanFactory beanFactory = new DefaultListableBeanFactory();
beanFactory.registerBeanDefinition("programmer", beanDefinitionBuilder.getBeanDefinition());
Programmer programmer = (Programmer) beanFactory.getBean("programmer");
programmer.showMaterial();
}
}
运行结果如下:
在使用 XML 方式时一般是通过调用 ClassPathXmlApplicationContext 来注册 Bean 的,其构造函数可以传入具体的 XMl配置文件的路径,可以是一个或者多个,甚至还可以是通配符。在构造函数内部就会调用熟悉的 refresh 方法了。
深入 refresh 方法可以发现,在该方法中调用了 obtainFreshBeanFactory 方法来获取生成的 Bean,这个方法实际上是调用了抽象实现类 AbstractRefreshableApplicationContext 的 refreshBeanFactory 方法,该方法首先会先判断此时是否还有 beanFactory ,如果有的话会先销毁 beanFactory,然后再重新创建一个 BeanFactory(实际上是 DefaultListableBeanFactory 类型),最后会调用 loadBeanDefinitions 加载我们定义的 XMl 配置,方法使用的是 XmlBeanDefinitionReader 来读取的 XMl 配置,下面一起来深入的了解一下 Spring 生成 Bean 的过程。
2. 创建 Bean 的过程
首先我们看看 BeanFactory 类图,如下所示:
Bean 的整体创建流程如下所示:
3. 总结
本文简要的讲述了 Spring 创建 Bean 的主要流程,还有许多细节的地方需要深入研读源码才能了解,在这里先给自己一个小目标,后续会自己实现一个简易版本的 Spring(IOC、AOP),预知后事如何,请看下篇博文。。。
Spring 是如何造出一个 Bean 的的更多相关文章
- spring 在容器中一个bean依赖另一个bean 需要通过ref方式注入进去 通过构造器 或property
spring 在容器中一个bean依赖另一个bean 需要通过ref方式注入进去 通过构造器 或property
- 在spring中如何生成一个bean (一个对象,比如jedis的连接池对象)【我】
在spring中,要想生成一个单例对象(比如jedis的连接池对象) 方法1: 在 spring中用 bean 标签生成(反正就是让spring生成并管理单例的对象) 方法2: 把要生成的单例对象类, ...
- Spring的几种注入bean的方式
在Spring容器中为一个bean配置依赖注入有三种方式: · 使用属性的setter方法注入 这是最常用的方式: · 使用构造器注入: · 使用Filed注入(用于注解方式). 使用属性的se ...
- 《Spring实战》系列之Bean的装配-Days02
2.1 回顾 对于我第一天在bean的装配中写的,是一些基本的语法或者是Spring本身的一些规定,但是我没有对此进行深究.接下来就让我们仔细的讨论一下细节问题.和传统的类的定义和方法的调用做一些比较 ...
- 【spring set注入 注入集合】 使用set注入的方式注入List集合和Map集合/将一个bean注入另一个Bean
Dao层代码: package com.it.dao; public interface SayHell { public void sayHello(); } Dao的Impl实现层: packag ...
- Spring学习笔记--声明一个简单的Bean
spring依赖的maven dependencyhttp://mvnrepository.com/artifact/org.springframework 在pom.xml中添加如下依赖: < ...
- 品Spring:真没想到,三十步才能完成一个bean实例的创建
在容器启动快完成时,会把所有的单例bean进行实例化,也可以叫做预先实例化. 这样做的好处之一是,可以及早地发现问题,及早的抛出异常,及早地解决掉. 本文就来看下整个的实例化过程.其实还是比较繁琐的. ...
- spring中如何向一个单例bean中注入非单例bean
看到这个题目相信很多小伙伴都是懵懵的,平时我们的做法大都是下面的操作 @Component public class People{ @Autowired private Man man; } 这里如 ...
- Spring boot 将配置文件属性注入到一个bean中
现在要做的就是将如下配置文件中的内容注入到一个bean 名为Properties中. Redis.properties配置文件中的内容如下: Properties java bean中代码如下,注意注 ...
- Spring IoC源码解析——Bean的创建和初始化
Spring介绍 Spring(http://spring.io/)是一个轻量级的Java 开发框架,同时也是轻量级的IoC和AOP的容器框架,主要是针对JavaBean的生命周期进行管理的轻量级容器 ...
随机推荐
- CPU上下文切换 CPU的调度策略
CPU上下文切换 就是先把前一个任务的CPU上下文(也就是CPU寄存器和程序计数器)保存起来,然后加载新任务的上下文,到这些寄存器和程序计数器,最后再跳转到程序计数器所指的新位置,运行新任务. 根据任 ...
- Linux编写Shell脚本获取指定目录下所有文件并处理
Linux编写Shell脚本获取指定目录下所有文件进行处理并保存到新目录 #!/bin/bash app_name="shell" path="/dir" #原 ...
- archlinux xfce 设置fcitx5中文输入法详细教程
1.安装fcitx5 sudo pacman -S fcitx5-im fcitx5-chinese-addons fcitx5-pinyin-zhwiki fcitx5-im 包含fcitx5的一些 ...
- Jmeter线程组-上
线程组 线程组作为JMeter测试计划的核心组件之一,对于模拟并发用户的行为至关重要.线程组元件是整个测试计划的入口,所有的取样器和控制器必须放置在线程组下. 可以将线程组视为一个虚拟用户池,其中每个 ...
- Jetpack Compose(4)——重组
目录 一.状态变化 1.1 状态变化是什么 1.2 mutableStateListOf 和 mutableStateMapOf 二.重组的特性 2.1 Composable 重组是智能的 2.2 C ...
- #线段树、树状数组#D 筹备计划
分析 首先这个位置应该是带权中位数\((\geq \frac{sum+1}{2}(奇数要加一,WA了几次了))\),但是既然有这个选择的限制, 那么要用线段树求出可选择的前驱和后继,然后用树状数组计算 ...
- 如何成为10x倍工程师
10倍效率 +10x 的工程师很难找,但是 -10x 工程师是存在的. 所谓 -10x 工程师,就是每周要浪费团队 400 个小时的工程师. 他有以下特征: 创造无效的繁忙工作,比如演示文稿.图表.工 ...
- Git分支教程:详解分支创建、合并、删除等操作
Git是一种强大的分布式版本控制系统,它的分支功能使得团队协作和代码管理变得更加灵活和高效.分支可以让开发人员在不影响主线开发的情况下进行并行开发和实验性工作.本篇博客将详解Git分支的创建.合并.删 ...
- Docker学习路线12:开发者体验
到目前为止,我们只讨论了使用Docker来部署应用程序.然而,Docker也是一个极好的用于开发应用程序的工具.可以采用一些不同的建议来改善开发体验. 在应用程序中使用docker-compose以方 ...
- HarmonyOS属性动画开发示例(ArkTS)
介绍 利用ArkUI组件不仅可以实现属性变化引起的属性动画,也可以实现父组件状态变化引起子组件产生动画效果,这种动画为显式动画.效果如图所示: 相关概念 显式动画:提供全局animateTo显式动 ...