【Spring】三级缓存

目录标题

  • 触发所有未加载的实例a - 开始
  • getBean( doGetBean) - 获取单例bean
    • getSingleton() - 获取单例bean
    • createBean(doCreateBean) - 创建bean
      • createBeanInstance - 创建并返回bean
      • addSingletonFactory -放三级缓存
      • populateBean - 属性设值
        • applyPropertyValues - 设值属性
          • 初始化b开始
            • 从缓存中获取b
            • 创建bean-b实例
            • 提前暴露bean-b
            • b 属性设值
            • 放入一级缓存里面
          • 初始化b结束
          • image.png
      • addSingleton- 放一级缓存
  • 触发所有未加载的实例a - 结束
  • 触发所有未加载的实例b - 开始
  • 触发所有未加载的实例b - 结束

结合文章:循环依赖

测试代码如下:

public class A {
    private B b;

    public B getB() {
        return b;
    }

    public void setB(B b) {
        this.b = b;
    }

    public A() {
        System.out.println("---A created success");
    }
}
public class B {
    private A a;

    public A getA() {
        return a;
    }

    public void setA(A a) {
        this.a = a;
    }
    
    public B() {
        System.out.println("---B created success");

    }
}
public class ClientSpringContainer {


    public static void main(String[] args) {
        sampleDemo();
    }


    private static void sampleDemo() {
        ApplicationContext context = new ClassPathXmlApplicationContext("applicationContext.xml");
        A a = context.getBean("a", A.class);
        B b = context.getBean("b", B.class);
    }
}

执行refresh 方法
image.png
执行finishBeanFactoryInitialization 方法
image.png
执行preInstantiateSingletons 方法
image.png

触发所有未加载的实例a - 开始

image.png

getBean( doGetBean) - 获取单例bean

实际上就是通过 doGetBean ,先进行 bean-a 的初始化

getSingleton() - 获取单例bean

去缓存查看时候有 bean - a
image.png
实际上就是通过双重校验锁,去查看一级缓存中是否有 bean-a 并且没有在创建中 ,所以就返回 null
image.png
由于返回了 null ,则 继续往下走,去创建bean-a实例
image.png由于我们 bean-a 是单例的,所以就执行下面的语句块
image.png
再进入内部,发现返回的其实就是执行的就是传参的 createBean(beanName, mbd, args)
image.png
image.png

createBean(doCreateBean) - 创建bean

image.png

createBeanInstance - 创建并返回bean

先执行的是 createBeanInstance,里面是通过构造函数去创建一个bean实例
image.png

addSingletonFactory -放三级缓存

继续,需要暴露出这个bean-a到 三级缓存中,此时我们是有了bean-a的实例:A@2321
image.png

  1. 往三级缓存中放入 a - lamdba@2337
  2. 删除二级缓存

image.png

populateBean - 属性设值

image.png

  1. 先解析看看bean -a 需要那些属性
  2. 在进行设值

image.png

applyPropertyValues - 设值属性

image.png
resolveValueIfNecessary 实际上调用了 resolveReference
image.png
resolveReference 也是通过beanFactory 中获取bean
image.png


初始化b开始

而由于此时工厂里面没有bean-b,没有进行初始化
实际上是调用了doGetBean,又开始了 实例化 bean-b的过程,getBean
image.png

从缓存中获取b

查看一级缓存中是否存在bean-b,返回null
image.png

创建bean-b实例

相当于重复了createBean
由于我们在前一步返回的是null,所以就去执行else语句块的内容
image.png
image.png
常见的是单例bean,进入方法,执行的是createBean方法
image.png
image.png
image.png
调用了doCreateBean方法
image.png
createBeanInstance , B@2627
image.png

提前暴露bean-b

addSingletonFactory
image.png
三级缓存中放入 b - lambda@2644
image.png
此处我们也可以看到在三级缓存中有两个
image.png

b 属性设值

调用 populateBean
image.png
applyPropertyValues
image.png
这里我们可以看到 b 是需要a的(符合我们 前面的需求:a b 互相引用)
image.png
applyPropertyValues 里面又调用了 resolveValueIfNecessary
image.png
resolveValueIfNecessary 又调用了 resolveReference
image.png
resolveReference 调用了 beanFactory.getBean,getBean又是通过doGetBean去获取
image.png
image.png
image.png
先从一级缓存中去获取bean-a ,返回null
image.png
由于一级缓存中没有且a在创建中,执行if语句块
image.png
查看二级缓存中是否有a,没有,执行if 语句块
image.png
再从三级缓存中去获取a, 这里是能够获取从三级缓存中获取到的点击访问
image.png
image.png

  1. a ,三级缓存中的实例 lambda@2337 对应的 实例 A@2321
  2. 放入二级缓存中去 A@2321,此时就已经把a放入二级缓存里面了
  3. 删除三级缓存的内容
  4. 返回缓存中的实例 A@2321

image.png
由于能够从三级缓存中去获取到半成品a,A@2321
image.png
doGetBean 返回 从三级缓存中获取到的A@2321,所以执行if语句快,发现最后返回的是bean,bean是通过getObjectForBeanInstance 去获取的
image.png
getObjectForBeanInstance,又去调用了 super.getObjectForBeanInstance
image.png
getObjectForBeanInstance 就是返回了 A@2321
image.png
doGetBean 结束了,返回了b需要的属性 a(虽然是半成品 A@2321)
image.png
resolveReference 结束,返回 A@2321
image.png
resolveValueIfNecessary结束,返回 A@2321
image.png
退回到applyPropertyValues
image.png
完成属性b设值a( A@2321)
image.png
此时populateBean
(回顾:此时就完成了b的setter注入a),完成了b初始化
image.png
继续,由于在前面,bean-b进行了提前暴露,执行if语句块,所以我们这次flase(意味不需要提前暴露了)
image.png
执行getSingleton, 由于从一级缓存中获取不到,且b在创建中,执行if语句块
image.png
从二级缓存中获取bean-b(肯定是没有的),下图中可以看到我们在二级缓存中也获取不到bean-b,并且不需要提前暴露了,所以不需要执行if语句块
image.png
所以就直接返回null
image.png
返回B@2627
image.png
此时我们就结束了对createBean ,并返回B@2627
image.png
于是我们就能够从三级缓存中去获取到b了
返回b 的 getSingleton,此时就是 B@2627
image.png

放入一级缓存里面

而且,此时我们这个是新创建的bean ,因此 newSingleton = true,执行addSingleton
image.png

  1. 放入一级缓存 b - B@2627
  2. 移除三级缓存
  3. 移除二级缓存

image.png
getSingleton 结束,获取到了 b实例 B@2627
image.png
doGetBean 结束,返回 b实例 B@2627,也就是这一步结束了完成b的初始化
image.png

初始化b结束

那么,接下来,继续完成a的初始化。
resolveReference 结束,返回 b实例 B@2627
image.png
resolveValueIfNecessary结束,返回 b实例 B@2627
image.png
设值a的属性b
image.png
populateBean 结束,完成属性赋值
image.png


初始化a ,A@2321
image.png
前面我们说过我们已经提前暴露了a到三级缓存池里面放到三级缓存池
image.png
执行 getSingleton ,由于
一级缓存里面没有a,且a在创建中,所以 执行if语句块
image.png
由于我们在二级缓存里面能够找到a,if语句就不执行了,因为已经完成了对a的放入二级缓存池

image.png

返回二级缓存池中的对象,A@2321
image.png
于是我们就有了早期暴露对象exposedObject,A@2321,doCreateBean结束
image.png
createBean 结束
image.png
doGetBean结束,返回到getSingleton
image.png
由于这是一个新创建的bean,newSingleton = true,执行addSingleton
image.png

addSingleton- 放一级缓存

  1. a 放入一级缓存:a - A@2321
  2. 移除二级缓存
  3. 移除三级缓存

image.png
getSingleton 结束
image.png
doGetBean 结束
image.png

触发所有未加载的实例a - 结束

image.png

触发所有未加载的实例b - 开始

image.png
image.png
执行getSingleton
image.png
此时,我们的b已经放入了一级缓存了哦,此处就已经完成了b放入一级缓存池,不执行if语句块,返回B@2627
image.png
而我们getGetBean的返回对象bean,就是getSingleton 返回的对象B@2627
image.png

触发所有未加载的实例b - 结束

image.png


preInstantiateSingletons 后面一个循环的语句块,由于这次我们关注的是“循环依赖”,就不着重分析这块 就直接过了
image.png
finishBeanFactoryInitialization 执行结束
image.png
refresh 执行结束
image.png
ClassPathXmlApplicationContext 执行结束
image.png


断点数量如下:
image.png


流程示例图
在这里插入图片描述

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/398450.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

CVE-2024-24565 CrateDB数据库任意文件读取漏洞

目录 前言 简介 ​编辑 环境搭建 漏洞复现 前言 本次介绍的漏洞不同与以往, 本次洞更多是适用于利用数据库提权。 利用数据库的导入导出数据的功能,我们往往可以将内部的一些敏感文件如/etc/passwd导入到数据库进行查看。也可以将数据导入到一些特…

Android 基础技术——Framework

笔者希望做一个系列,整理 Android 基础技术,本章是关于 Framework 简述 Android 系统启动流程 当按电源键触发开机,首先会从 ROM 中预定义的地方加载引导程序 BootLoader 到 RAM 中,并执行 BootLoader 程序启动 Linux Kernel&…

见微知著:数据可视化助力数字化时代决策智慧

在数字化时代的浪潮中,数据可视化显然是推动数字化进程不可或缺的利器。通过将复杂的数据转化为直观的图形和图表,数据可视化为企业和组织提供了更清晰、更有效的方式来理解和应用大量的数字信息。下面我就以可视化从业者的角度,来简单聊聊这…

Spring Boot项目怎么对System.setProperty(key, value)设置的属性进行读取加解密

一、前言 之前我写过一篇文章使用SM4国密加密算法对Spring Boot项目数据库连接信息以及yaml文件配置属性进行加密配置(读取时自动解密),对Spring Boot项目的属性读取时进行加解密,但是没有说明对System.setProperty(key, value)设…

高性能三防平板有哪几款推荐的|亿道三防onerugged

我非常荣幸地向大家推荐几款高性能的三防平板电脑,这些产品都来自亿道三防onerugged系列。它们以其卓越的性能和出色的工艺设计,成为行业中备受瞩目的产品。 首先,让我们关注这些高性能三防平板所搭载的处理器。这些产品采用了基于Intel Cor…

opencv之cvScalar

CV_INLINE CvScalar cvScalar( double val0, double val1 CV_DEFAULT(0), double val2 CV_DEFAULT(0), double val3 CV_DEFAULT(0)) { CvScalar scalar; scalar.val[0] val0; scalar.val[1] val1; scalar.val[2] val2; scalar.val[3] val3; return scalar; } 该函数的…

【ArcGIS微课1000例】0104:二位面状数据转三维多面体(建筑物按高度拉伸)

文章目录 一、加载数据二、添加高度字段三、三维拉伸显示四、生成三维体数据五、注意事项一、加载数据 打开ArcScene,加载配套实验数据(0104.rar中的二维建筑物矢量数据,订阅专栏,获取专栏所有文章阅读权限及配套数据),如下图所示: 二、添加高度字段 本实验将二维数据…

k8s除了可以直接运行docker镜像之外,还可以运行什么? springboot项目打包成的压缩包可以直接运行在docker容器中吗?

Kubernetes(k8s)主要设计用于自动部署、扩展和管理容器化应用程序。虽然它与Docker容器最为密切相关,Kubernetes实际上是与容器运行时技术无关的,这意味着它不仅仅能够管理Docker容器。Kubernetes支持多种容器运行时,包…

潇洒郎:2024 IDEA、Pycharm获取最新激活码获取方式

IDEA获取最新激活码 https://idea.javatiku.cn/ 手机打开,看到验证码,30分钟有效,输入验证码 获取到最新激活码

Python Flask高级编程之RESTFul API前后端分离(学习笔记)

Flask-RESTful是一个强大的Python库,用于构建RESTful APIs。它建立在Flask框架之上,提供了一套简单易用的工具,可以帮助你快速地创建API接口。Flask-RESTful遵循REST原则,支持常见的HTTP请求方法,如GET、POST、PUT和DE…

PostgreSQL面试题-灿总题

面试题来源 :PostgreSQL学徒 PostgreSQL面试题集锦 已有的答案:Hehuyi_In 《PostgreSQL面试题集锦》学习与回答 一、MVCC 实现机制以及和 Oracle 的差异 ORACLE,MYSQL都是使用的UNDO来实现多版本并发控制,undo条目记录在从额外…

多维时序 | Matlab实现基于VMD-DBO-BiLSTM、VMD-BiLSTM、BiLSTM的多变量时间序列预测

多维时序 | Matlab实现基于VMD-DBO-BiLSTM、VMD-BiLSTM、BiLSTM的多变量时间序列预测 目录 多维时序 | Matlab实现基于VMD-DBO-BiLSTM、VMD-BiLSTM、BiLSTM的多变量时间序列预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 Matlab实现基于VMD-DBO-BiLSTM、VMD-BiLSTM、…

说一下JVM类加载机制?

Java中的所有类,都需要由类加载器装载到JVM中才能运行。类加载器本身也是一个类,而它的工作就是把class文件从硬盘读取到内存中。 在写程序的时候,我们几乎不需要关心类的加载,因为这些都是隐式装载的,除非我们有特殊…

HTB pwn Bad gradesRestaurant

Bad grades 没有开启ASLR 在功能2存在越界写,虽然有canary但是可以通过输入’-绕过。 输入数据可以利用python的struct解包成双精度浮点数,做了这些就是常规的ROP环节了。 exp from pwn import * def gd():gdb.attach(p)pause() def htd(hex_value):binary hex_value.to_b…

使用 Nginx 将部署多个子域名http服务重定向到https服务

首先在阿里云解析对应的子域名 打开自己的域名解析 点击添加记录 然后在nginx的配置文件下新增以下内容 # ssl证书地址,是你自己服务器上证书存放的地址 ssl_certificate /ssl/fullchain.cer; ssl_certificate_key /ssl/cert.key;#请按照以下协议配置 ssl_protocols…

JS基础(语法结构变量数据类型运算符流程控制)

JS基础(语法结构/变量/数据类型/运算符/流程控制) 目录 JS基础(语法结构/变量/数据类型/运算符/流程控制)什么是js?注释语法语法结构引入方式【1】script标签内部直接书写js代码【2】script标签src属性引入外部js代码 JS基础数据类型变量与常量变量的定义常量的定义…

OpenHarmony JS和TS三方组件使用指导

OpenHarmony JS和TS三方组件介绍 OpenHarmony JS和TS三方组件使用的是OpenHarmony静态共享包,即HAR(Harmony Archive),可以包含js/ts代码、c库、资源和配置文件。通过HAR,可以实现多个模块或者多个工程共享ArkUI组件、资源等相关代码。HAR不…

calcite在flink中的二次开发,介绍解析器与优化器

calcite 在flink中的二次开发 1 CodeGen2 flink 语法扩展2.1 在进行 Rule 规则匹配时,放开对 Distinct 的限制2.2下面附上一个 利用codegen来生成所需类的例子: 3 flink使用calcite 生成解析器FlinkSqlParserImpl3.1 FlinkSqlParserImpl 的生成3.1.1 fli…

[SwiftUI]启动页LaunchScreen.storyboard中适配状态栏加安全区域的高度

如下图,我有一个需求。在启动页(LaunchScreen.storyboard)和引导页(GuideView)的黑色背景上,使用了同一张正方形图片。要求从启动页切换到引导页时,这两张相同的图片的过渡要无缝衔接&#xff0…

redis的缓存穿透,缓存并发,缓存雪崩,缓存问题及解决方案

缓存穿透 问题原因 解决方案 缓存并发 缓存雪崩 缓存失效时间设置一致导致的。 解决方案: 1)方案一 2)方案二 如何设计一个缓存策略,缓存热点数据?