IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> Java知识库 -> 【Spring源码】Bean对象的创建和从容器中获取对象的过程 -> 正文阅读

[Java知识库]【Spring源码】Bean对象的创建和从容器中获取对象的过程



一、前期准备

1.1 环境依赖

<dependencies>
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-context</artifactId>
        <version>5.1.7.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-webmvc</artifactId>
        <version>5.1.7.RELEASE</version>
    </dependency>
</dependencies>

1.2 实体类

简单的User类,在测试过程中创建这个User类的对象。

public class User {
    private Integer id;
    private String name;

    public User() {
        System.out.println("创建了");
    }
}

1.3 applicationContext.xml

在applicationContext.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"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">

    <bean class="com.zqc.domain.User" id="user">

    </bean>
</beans>

1.4 测试代码

通过applicationContext.xml配置应用程序的上下文,在容器中创建User对象。

public class SpringDemo {
    public static void main(String[] args) {
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        User user = (User) context.getBean("user");
    }
}

二、探究过程

2.1 目标

目标:Spring容器中的Bean是何时创建的?为什么我们能从容器中获取到?

我们要带着上面的目标去读源码,只看跟我们目标相关的代码(Bean是何时创建的),不要在读源码的过程中迷失自我,始终要记得我们究竟要探究什么问题。遇到好奇或不懂的地方可以先记录下来,以后再去探究。

2.2 方法

如果调用User的无参构造器创建了User对象,那么控制台将打印【创建了】。

public class User {
    private Integer id;
    private String name;

    public User() {
        System.out.println("创建了");
    }
}

我们通过打断点Debug的方式,时刻观察控制台的输出分析是哪段代码执行结束后创建了User对象。

可以通过【Ctrl + 鼠标左键】查看某个方法的具体实现;如果通过【Ctrl + 鼠标左键】跳转到了接口的抽象方法上,此时,我们需要通过【Ctrl + Alt + 鼠标左键】跳转到具体的实现方法上。

一个接口可以有很多的实现类,但是在Debug的过程中我们无需担心不知道选择哪个实现类,因为程序已经运行了,实现类已经唯一确认,我们通过【Ctrl + Alt + 鼠标左键】就可以跳转到正确的实现类的方法上。

2.3 探究创建Bean对象过程

2.3.1 SpringDemo测试类

在这里插入图片描述
在执行完ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");这一行代码后,对象就已经创建好了。那么就说明在创建ClassPathXmlApplicationContext这个对象的过程中创建了Bean对象。

2.3.2 ClassPathXmlApplicationContext

在ClassPathXmlApplicationContext中重载了很多构造器,我们只看程序调用到的构造器。
在这里插入图片描述
this(new String[] {configLocation}, true, null);调用了另一个构造器
在这里插入图片描述

从xml文件中读取配置,通过parent(不用管它是什么)创建ClassPathXmlApplicationContext对象。

在Debug过程中,发现在执行完refresh()方法后就完成了bean对象的创建。暂时不需要理解这个构造器中其他方法的作用。

2.3.3 AbstractApplicationContext

🔶 refresh()方法

跳转到了AbstractApplicationContext类中的refresh()方法。

在Debug过程中,发现在执行完下面这行代码后创建了bean对象。

// Instantiate all remaining (non-lazy-init) singletons.
finishBeanFactoryInitialization(beanFactory);

从名字和注解上我们可以看出来,这个方法可以实例化所有非懒加载的单例对象

🔶 finishBeanFactoryInitialization()方法

跳转到了AbstractApplicationContext类中的finishBeanFactoryInitialization()方法。

在Debug过程中,发现在执行完下面这行代码后创建了bean对象。

// Instantiate all remaining (non-lazy-init) singletons.
beanFactory.preInstantiateSingletons();

从名字和注解上我们可以看出来,这个方法是通过bean工厂实例化所有非懒加载的单例对象

通过【Ctrl + Alt + 左键】跳转。

2.3.4 DefaultListableBeanFactory

preInstantiateSingletons()是ConfigurableListableBeanFactory接口中的方法,DefaultListableBeanFactory是ConfigurableListableBeanFactory接口实现类。

我们只需要看DefaultListableBeanFactory实现类中的preInstantiateSingletons方法即可。

通过for循环实例化所有非懒加载对象,不过这里只包含一个User对象。
在这里插入图片描述
上面的代码中,我们可以先了解一下BeanDefinition,他就是描述类详细信息的对象,其中包括我们了解的scope=”singleton”表示单例,lazyInit=false表示非懒加载。
在这里插入图片描述
① 通过if语句判断这个对象不是抽象类、是单例、非懒加载。(成立)

if (!bd.isAbstract() && bd.isSingleton() && !bd.isLazyInit()) {

② 通过if判断这个对象是不是FactoryBean。(不成立)

if (isFactoryBean(beanName))

在这里插入图片描述
所以最后来到了getBean(beanName)方法。

2.3.5 AbstractBeanFactory

🔶 getBean()方法

跳转到AbstractBeanFactory抽象类的getBean方法。

@Override
public Object getBean(String name) throws BeansException {
	return doGetBean(name, null, null, false);
}

这个方法由调用了doGetBean(name, null, null, false)

🔶 doGetBean()方法

在这里创建bean对象,通过lambda表达式和匿名内部类来实现的。
在这里插入图片描述
最终创建bean对象的代码为createBean(beanName, mbd, args),调用这个createBean()方法的函数其实在getSingleton()当中。因为这里通过lambda表达式和匿名内部类的方式创建了对象传入了getSingleton()方法中。

虽然createBean(beanName, mbd, args)创建了bean对象,但是我们还是先去看看getSingleton()

3.3.6 DefaultSingletonBeanRegistry

下面来看看getSingleton()方法,注意他传入了singletonFactory对象
在这里插入图片描述
创建bean对象的方法为singletonFactory.getObject(),而这个singletonFactory是传入的参数,也就是上面提到的lambda表示和匿名内部类创建的对象。这个对象里面就包含了getObject方法。

不太懂的朋友可以看看lambda表示和函数式接口。
这个ObjectFactory就是函数式接口,里面包含了getObject方法。
在这里插入图片描述

所以这段代码就是实现了3.3.5节中的:
在这里插入图片描述

3.3.7 AbstractAutowireCapableBeanFactory

🔶 createBean()方法

下面跳转到createBean()方法中:
在这里插入图片描述
beanName:类的名称
mbd:类的定义信息
在这里插入图片描述

通过调用doCreateBean(beanName, mbdToUse, args)创建bean对象。

🔶 doCreateBean()方法
在这里插入图片描述
通过调用createBeanInstance(beanName, mbd, args)创建bean对象。

🔶 createBeanInstance()方法

在这个方法中经过一系列的分析,使用无参构造器创建对象。
在这里插入图片描述
🔶 instantiateBean()方法
在这里插入图片描述
通过调用getInstantiationStrategy().instantiate(mbd, beanName, parent)创建bean对象。

3.3.8 SimpleInstantiationStrategy

获取空参构造器,通过Bean工具实例化类对象。
在这里插入图片描述

3.3.9 BeanUtils

BeanUtils:JavaBeans的静态方法,用于初始化bean、检查bean的属性类型、复制bean属性等。

通过传入构造器ctor和所需的参数args创建bean对象。最终通过ctor.newInstance(args)创建了对象并返回。
在这里插入图片描述

2.4 探究Bean对象加入容器过程

🔶 getSingleton()方法

我们回到【3.3.6节】的DefaultSingletonBeanRegistry的getSingleton()方法中:
在这里插入图片描述
我们知道了singletonObject = singletonFactory.getObject()创建了bean对象。

此时对象就是singletonObject,同时令newSingleton = true,再往下看:
在这里插入图片描述

if (newSingleton)成立,将执行addSingleton(beanName, singletonObject)

🔶 addSingleton()方法

将单例对象加入到工厂的单例缓存singletonObjects
在这里插入图片描述
原来这里面已经包含了五个对象:
在这里插入图片描述
这段代码运行结束后,新增一个user对象:
在这里插入图片描述

2.5 从容器中获取Bean对象的过程

从刚刚创建的容器中获取user对象。
在这里插入图片描述
🔶 AbstractApplicaitonContext的getBean()方法
在这里插入图片描述
🔶 AbstractBeanFactory的getBean()方法
在这里插入图片描述
又回到了doGetBean方法,在创建bean对象的过程中也用到了doGetBean方法。

🔶 AbstractBeanFactory的doGetBean()方法

这时候我们就可以通过方法getSingleton(beanName);获取bean对象,最后返回这个bean对象。
在这里插入图片描述
🔶 DefaultSingletonBeanRegistry的getSingleton()方法

下面来看看getSingleton()方法:
在这里插入图片描述

2.6 总结

非懒加载的单实例bean会在容器创建的时候创建。容器内部会创建一个beanFactory,使用beanFactory的doGetBean方法来进行创建,并且在创建后会把bean放入一个单例bean的map集合(singletonObjects)中存储。key就是我们配置的bean的名称。

我们调用容器getBean()方法来获取对象的时候,其实也是调用了doGetBean方法。就会从对应的集合中获取到之前创建的对象。

2.7 新的疑问

RootBeanDefination是什么?如何通过bean名获取对应的RootBeanDefinition对象的?

如果是FactoryBean是怎么创建的?

  Java知识库 最新文章
计算距离春节还有多长时间
系统开发系列 之WebService(spring框架+ma
springBoot+Cache(自定义有效时间配置)
SpringBoot整合mybatis实现增删改查、分页查
spring教程
SpringBoot+Vue实现美食交流网站的设计与实
虚拟机内存结构以及虚拟机中销毁和新建对象
SpringMVC---原理
小李同学: Java如何按多个字段分组
打印票据--java
上一篇文章      下一篇文章      查看所有文章
加:2022-05-25 11:32:41  更:2022-05-25 11:33:43 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2024年11日历 -2024/11/23 20:01:54-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码