首页 > 编程语言 >【框架源码】Spring源码解析之Bean创建源码流程

【框架源码】Spring源码解析之Bean创建源码流程

时间:2023-06-17 21:04:30浏览次数:63  
标签:初始化 缓存 对象 Spring bean Bean 源码 单例


【框架源码】Spring源码解析之Bean创建源码流程_spring

问题:Spring中是如何初始化单例bean的?

我们都知道Spring解析xml文件描述成BeanDefinition,解析BeanDefinition最后创建Bean将Bean放入单例池中,那么Spring在创建Bean的这个过程都做了什么。

Spring核心方法refresh()中最最重要的一个方法 finishBeanFactoryInitialization() 方法,该方法负责初始化所有的单例bean。

finishBeanFactoryInitialization()方法位于refresh()中第11步。

【框架源码】Spring源码解析之Bean创建源码流程_后端_02

走到这一步的时候,Spring容器中所有的BeanFactory都已经实例化完成了,也就是实现BeanFactoryPostProcessor接口的 Bean 都已经初始化完成了。剩下的就是初始化singleton beans,在我们的业务bean中大多数都是单例的,finishBeanFactoryInitialization这一步就是去实例化单例的并且没有设置懒加载的Bean。

Spring会在finishBeanFactoryInitialization这个方法里面初始化所有的singleton bean

Ok,我们先来看一下finishBeanFactoryInitialization方法内部的逻辑。这个方法的核心就在于完成BeanFactory的配置。该阶段完成了上下文的实例化,包含所有单例Bean对象已经实例化。

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_03

核心在于 preInstantiateSingletons() 方法,preInstantiateSingletons方法主要任务是进行初始化,在初始化前同样是一系列判断,如,是否是懒加载的,是否是一个factorybean(一个特别的bean,负责工厂创建的bean),最后调用getBean()方法。

【框架源码】Spring源码解析之Bean创建源码流程_spring_04

【框架源码】Spring源码解析之Bean创建源码流程_spring_05

【框架源码】Spring源码解析之Bean创建源码流程_后端_06

注释中提到的SmartInitializingSingleton接口,是让bean初始化后做一些操作。

【框架源码】Spring源码解析之Bean创建源码流程_java_07


【框架源码】Spring源码解析之Bean创建源码流程_后端_08

OK,那么主要的方法还是getBean()方法,getBean()方法的作用就是加载、实例化Bean。方法内部调用了doGetBean(),我们直接看**doGetBean()**方法内部。

【框架源码】Spring源码解析之Bean创建源码流程_java_09

【框架源码】Spring源码解析之Bean创建源码流程_后端_10

【框架源码】Spring源码解析之Bean创建源码流程_spring_11

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_12

【框架源码】Spring源码解析之Bean创建源码流程_spring_13

【框架源码】Spring源码解析之Bean创建源码流程_后端_14

【框架源码】Spring源码解析之Bean创建源码流程_后端_15

【框架源码】Spring源码解析之Bean创建源码流程_后端_16

代码很多,我们主要看createBean()方法。

【框架源码】Spring源码解析之Bean创建源码流程_后端_17

【框架源码】Spring源码解析之Bean创建源码流程_初始化_18

【框架源码】Spring源码解析之Bean创建源码流程_后端_19

我们继续往doCreateBean 这个方法里面看。

【框架源码】Spring源码解析之Bean创建源码流程_spring_20

【框架源码】Spring源码解析之Bean创建源码流程_spring_21

【框架源码】Spring源码解析之Bean创建源码流程_初始化_22

【框架源码】Spring源码解析之Bean创建源码流程_spring_23

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_24

【框架源码】Spring源码解析之Bean创建源码流程_spring_25

方法很多,我们主需要关注三个方法即可。

  • createBeanInstance:实例化,其实也就是调用对象的构造方法实例化对象
  • populateBean:填充属性,这一步主要是多bean的依赖属性进行填充
  • initializeBean:调用spring xml中的init 方法。

从上面讲述的单例bean初始化步骤我们可以知道,循环依赖主要发生在第一、第二步。也就是构造器循环依赖和field循环依赖。

那么我们要解决循环引用也应该从初始化过程着手,对于单例来说,在Spring容器整个生命周期内,有且只有一个对象,所以很容易想到这个对象应该存在Cache中,Spring为了解决单例的循环依赖问题,使用了三级缓存。

什么是Spring的三级缓存?

Spring的IOC容器里面的三级缓存都是Map结构。

  • 一级缓存(成熟的bean)
  • singletonObjects 单例池 ,存放完全初始化好的 bean,从该缓存中取出的 bean 可以直接使用
  • 二级缓存
  • earlySingletonObjects 提前曝光的单例对象的cache,存放原始的 bean 对象(尚未填充属性),用于解决循环依赖
  • 三级缓存
  • singletonFactories 单例对象工厂的cache,存放 bean 工厂对象,用于解决循环依赖

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_26

OK,了解完三级缓存我们再来看下 getSingleton() 这个方法。这个方法主要是用于从单例池中获取指定名称的单例Bean实例,方法内部实现了三级缓存查找机制,通过三级查找的机制来获取指定名称的单例Bean实例对象,同时该方法会使用同步代码块保证多线程环境下的线程安全性。

【框架源码】Spring源码解析之Bean创建源码流程_java_27

OK,那么到这里我们会有一个疑问,Spring为什么要用三级缓存来解决循环依赖的问题。

首先我们要明确一点,Spring可以解决setter的依赖注入,但是不能解决构造器的依赖注入。

假如我们现在有个A对象B对象A的某个field或者setter依赖了B的实例对象,同时B的某个field或者setter依赖了A的实例对象”这种循环依赖的情况。

A首先完成了初始化的第一步,并且将自己提前曝光到singletonFactories中,此时进行初始化的第二步,发现自己依赖对象B,此时就尝试去get(B),发现B还没有被create,所以走create流程,B在初始化第一步的时候发现自己依赖了对象A,于是尝试get(A),尝试一级缓存singletonObjects(肯定没有,因为A还没初始化完全),尝试二级缓存earlySingletonObjects(也没有),尝试三级缓存singletonFactories,由于A通过ObjectFactory将自己提前曝光了,所以B能够通过ObjectFactory.getObject拿到A对象(虽然A还没有初始化完全,但是总比没有好呀),B拿到A对象后顺利完成了初始化阶段1、2、3,完全初始化之后将自己放入到一级缓存singletonObjects中。

此时返回A中,A此时能拿到B的对象顺利完成自己的初始化阶段2、3,最终A也完成了初始化,进去了一级缓存singletonObjects中,而且更加幸运的是,由于B拿到了A的对象引用,所以B现在hold住的A对象完成了初始化。

【框架源码】Spring源码解析之Bean创建源码流程_java_28

知道了这个原理时候,肯定就知道为啥Spring不能解决"A的构造方法中依赖了B的实例对象,同时B的构造方法中依赖了A的实例对象"这类问题啦!因为加入singletonFactories三级缓存的前提是执行了构造器,所以构造器的循环依赖没法解决。

下面一个案例带大家体验下构造器注入和set注入的演示

编写类A类B

public class A {
    private B b;
    public A() {}
    public B getB() {
        return b;
    }
    public void setB(B b) {
        this.b = b;
    }
    public void method(){
        System.out.println("A方法调用");
    }
}
public class B {
    private A a;
    public B(){}
    public A getA() {
        return a;
    }
    public void setA(A a) {
        this.a = a;
    }
    public void method(){
        System.out.println("B方法调用");
    }
}

编写xml

<bean id="b" class="com.lixiang.demo.B">
        <property name="a" ref="a"></property>
    </bean>
    <bean id="a" class="com.lixiang.demo.A">
        <property name="b" ref="b"></property>
    </bean>

测试

【框架源码】Spring源码解析之Bean创建源码流程_java_29

修改xml配置

<bean id="b" class="com.lixiang.demo.B">
        <!--将a改成用构造器注入-->
        <constructor-arg name="a" ref="a"></constructor-arg>
    </bean>
    <bean id="a" class="com.lixiang.demo.A">
        <property name="b" ref="b"></property>
    </bean>

代码调整

public class B {
    private A a;
    public B(A a){
        this.a = a;
    }
    public A getA() {
        return a;
    }
    public void setA(A a) {
        this.a = a;
    }
    public void method(){
        System.out.println("B方法调用");
    }
}

测试

【框架源码】Spring源码解析之Bean创建源码流程_初始化_30

可以发现用构造器注入是发生异常的。

Spring引入了“提前暴露Bean”的机制,在创建A对象时,会先创建一个A的空对象并将其添加到缓存池中。

即“提前暴露Bean”,然后继续创建B对象,将其注入A对象中。在创建B对象时,由于A对象已经在缓存池中,可以直接获取到A对象,接着将B对象注入到A对象中,完成Bean的初始化。

【框架源码】Spring源码解析之Bean创建源码流程_spring_31

好的,到现在整一个Bean的创建流程,就已经完成啦。我们在看一下以下三个方法的具体实现。

首先第一个就是 createBeanInstance() 方法

【框架源码】Spring源码解析之Bean创建源码流程_初始化_32

【框架源码】Spring源码解析之Bean创建源码流程_java_33

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_34

然后是 populateBean() 方法

【框架源码】Spring源码解析之Bean创建源码流程_spring_35

【框架源码】Spring源码解析之Bean创建源码流程_java_36

【框架源码】Spring源码解析之Bean创建源码流程_三级缓存_37

【框架源码】Spring源码解析之Bean创建源码流程_java_38

最后是 initializeBean() 方法

【框架源码】Spring源码解析之Bean创建源码流程_java_39

【框架源码】Spring源码解析之Bean创建源码流程_spring_40

ok,到这里Spring的bean的创建过程就已经梳理完成啦。

标签:初始化,缓存,对象,Spring,bean,Bean,源码,单例
From: https://blog.51cto.com/u_15646271/6506206

相关文章

  • Spring框架中的线程池
    Spring框架中的线程池使用Java的ExecutorService接口实现ExecutorService是Java提供的用于管理线程池的高级工具。下面是在Spring框架中使用线程池的一般步骤:导入所需的依赖首先,确保你的项目中包含了使用线程池所需的依赖。通常情况下,你可以使用SpringBoot来创建项目,它会自动包含......
  • 2、spring
    1、简介springframework是其他spring框架的基础如springboot、springcloud2、Ioc  ......
  • SpringBoot整合ActiveMQ
    第一步: 第二步: 第三步: 下面如有需要才使用 ......
  • Springboot整合mongodb
    入门案例创建工程,导入依赖导入依赖点击查看代码<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-mongodb</artifactId><version>2.3.9.RELEASE</version>......
  • Spring Cloud 五大组件【Spring Cloud】
    组件名称作用Cloud体系阿里巴巴体系注册中心服务治理eurekanacos负载均衡集群负载confignacos配置中心配置管理Ribbonnacos网关请求的出入口GatewayGateway熔断器服务安全Hystrixsentinel......
  • 源码泄露+bak备份泄露+vim泄露+.DS_Store(mas迁移泄露)
    源码泄露+bak备份泄露+vim泄露+.DS_Store(mas迁移泄露)1.源码泄露web网站源码打包在web目录下造成泄露,通常以压缩包方式存在,如.zip、.rar、.tar、.tar.gz等,常见命名方式为网站名,www.网站名,backup+网站名等简单入门题目扫描到压缩包文件进行下载,找到对应文件,查看是否有flag,如果没......
  • 开源数字药店系统源码:打造高效的医药销售平台
    作为医药销售的全新解决方案,数字药店系统源码能够为医药企业提供更高效的销售解决方案,提高企业的竞争力。本文将详细介绍开源数字药店系统源码的特点和优势,以及如何打造高效的医药销售平台。一、开源数字药店系统源码的特点1. 功能丰富具有完善的功能,包括商品管理、订单管理、客户......
  • SpringBoot--MQ不生效
    Cannotresolveconfigurationproperty'rabbitmq.username'rabbitmq:username:adminpassword:adminvirtual-host:test_datalistener:simple:#表示消费者消费成功消息以后需要手工的进行签收(ack确认),默认为autoacknowledge-mode:manualprefet......
  • springboot @Bean自动注册
    这个注解,可以注册Bean到spring容器中@BeanpublicXXXBeanxxxBean(){returnnewXXXBean();}这个注解也可以用在void方法上,用在在spring容器启动后固定执行某个代码逻辑:@BeanpublicvoidxxxHandler(){System.out.println("我想容器启动后执行一次某个代......
  • SpringBoot学习笔记
    新建SpringBoot项目阿里云地址:https://start.aliyun.com异常消息处理//1.自定义异常类,继承RuntimeExceptionpublicclassMyExceptionextendsRuntimeException{publicMyException(){}}//2.定义全局异常类@RestControllerAdvicepublicclassGloabExcept......