手写spring IOC底层源码来模拟spring如何利用多级缓存解决循环依赖的问题

在文章开始之前,先来看一张spring IOC加载过程的脑图吧

Spring IOC的加载过程

首先,当我们去new了一个applicationContext,它底层呢就会把我们配置的bean进行扫描,然后创建成一个一个的beanDefinition放在我们的beanDefinitionMap中,此时就有了一切创造bean的原料信息,然后就会去循环beanDefinition,去调用beanfactory.getBean方法,先尝试在一级缓存中获取,获取不到呢就会进行创建,先进行实例化,然后进行依赖注入,最后初始化,放入到一级缓存中.

手写源码

package cn.edu.hunau;

import cn.edu.hunau.service.AService;
import cn.edu.hunau.service.BService;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.RootBeanDefinition;

import java.lang.reflect.Field;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @Author SuJ
 * @Date 2024 04 12 15 13.
 * 手写spring IOC底层源码来模拟spring如何利用多级缓存解决循环依赖的问题。
 **/
public class SuJApplicationContext {
    private Map<String, BeanDefinition> beanDefinitionMap = new LinkedHashMap<>();
    // 一级缓存 单例池
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();


    public SuJApplicationContext() throws Exception{
        // 加载ioc容器  创建所有的bean
        refersh();

        finishBeanFactoryInitialization();
    }

    //一个个的创建bean
    private void finishBeanFactoryInitialization() {
        //循环所有的beanDefinition
        beanDefinitionMap.keySet().forEach(
                beanName -> {
                    try {
                        getBean(beanName);
                    } catch (InstantiationException e) {
                        throw new RuntimeException(e);
                    } catch (IllegalAccessException e) {
                        throw new RuntimeException(e);
                    }
                }
        );
    }

    private Object getBean(String beanName) throws InstantiationException, IllegalAccessException {
        // 1.尝试在一级缓存中获取
        Object bean = getSingleton(beanName);
        //如果存在 直接放回
        if(bean != null){
            return bean;
        }

        // 2.创建  ---> 实例化
        RootBeanDefinition beanDefinition = (RootBeanDefinition)beanDefinitionMap.get(beanName);
        Class<?> beanClass = beanDefinition.getBeanClass();
        bean = beanClass.newInstance();

        //3. 依赖注入
        for (Field declaredField : beanClass.getDeclaredFields()){
            //当前属性有注解
            if(declaredField.getAnnotation(Autowired.class) != null){
                String name = declaredField.getName();
                Object dependBean = getBean(name);

                declaredField.setAccessible(true);
                declaredField.set(bean, dependBean);
            }
        }

        //4.初始化
//if(bean instanceof InitializingBean){
//    ((InitializingBean)bean).afterPropertiesSet();
//}

        //5.放入一级缓存
        singletonObjects.put(beanName, bean);
        return bean;

    }
//获取单例池中的bean
    private Object getSingleton(String beanName) {
        if(singletonObjects.containsKey(beanName)){
            return singletonObjects.get(beanName);
        }
        return null;
    }

    //ioc容器加载
    public void refersh() throws Exception{
        //1.解析配置 支持BeanDefinition
        loadBeanDefinitions();
    }

    /**
     *
     * 根据配置信息创建BeanDefinition  底层是通过解析配置类注册beandefiniton
     */
    private void loadBeanDefinitions(){

        // 创建A BeanDefinition
        RootBeanDefinition aBeanDefinition = new RootBeanDefinition(AService.class);
        //创建B BeanDefinition
        RootBeanDefinition bBeanDefinition = new RootBeanDefinition(BService.class);

        beanDefinitionMap.put("aService",aBeanDefinition);
        beanDefinitionMap.put("bService",bBeanDefinition);
    }
}

当我们手写完IOC容器的创建过程,会发现其实在一级缓存就可以解决循环依赖的问题,只需要增加一行代码。

我们可以发现程序正常执行

那为什么spring的设计人员不采取这种方式,而是要通过三级缓存来解决循环依赖的问题呢?

这是因为只通过一级缓存来解决循环依赖问题会造成线程安全问题,例如线程1先实例化A,直接放到一级缓存,这时线程2从一级缓存中获取到了实例,调用B实例的方法,由于没有进行依赖注入,我们的B实例为null,会造成空指针异常。

为了解决这个问题,我们引入了二级缓存,专门用于存储不完整的bean,使用二级缓存获取到的bean作为出口,并且将临界资源锁住(这里借用了单例模式的思想),果然解决了线程安全的问题。

package cn.edu.hunau;

import cn.edu.hunau.service.AService;
import cn.edu.hunau.service.BService;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.RootBeanDefinition;

import java.lang.reflect.Field;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @Author SuJ
 * @Date 2024 04 12 15 13.
 * 手写spring IOC底层源码来模拟spring如何利用多级缓存解决循环依赖的问题。
 **/
public class SuJApplicationContext {
    private Map<String, BeanDefinition> beanDefinitionMap = new LinkedHashMap<>();
    // 一级缓存 单例池
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();
    // 二级缓存 ---->  并发获取不完整bean------dcl
    private final Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>();


    public SuJApplicationContext() throws Exception {
        // 加载ioc容器  创建所有的bean
        refersh();

        finishBeanFactoryInitialization();
    }

    //一个个的创建bean
    private void finishBeanFactoryInitialization() {
        //循环所有的beanDefinition
        beanDefinitionMap.keySet().forEach(
                beanName -> {
                    try {
                        getBean(beanName);
                    } catch (InstantiationException e) {
                        throw new RuntimeException(e);
                    } catch (IllegalAccessException e) {
                        throw new RuntimeException(e);
                    }
                }
        );
    }

    private Object getBean(String beanName) throws InstantiationException, IllegalAccessException {
        // 1.尝试在一级缓存中获取
        Object bean = getSingleton(beanName);
        //如果存在 直接放回
        if (bean != null) {
            return bean;
        }
        synchronized (singletonObjects) {
            bean = getSingleton(beanName);
            //如果存在 直接返回
            if (bean != null) {
                return bean;
            }

            // 2.创建  ---> 实例化
            RootBeanDefinition beanDefinition = (RootBeanDefinition) beanDefinitionMap.get(beanName);
            Class<?> beanClass = beanDefinition.getBeanClass();
            bean = beanClass.newInstance();

            earlySingletonObjects.put(beanName, bean);

            //3. 依赖注入
            for (Field declaredField : beanClass.getDeclaredFields()) {
                //当前属性有注解
                if (declaredField.getAnnotation(Autowired.class) != null) {
                    String name = declaredField.getName();
                    Object dependBean = getBean(name);

                    declaredField.setAccessible(true);
                    declaredField.set(bean, dependBean);
                }
            }

            //4.初始化
//if(bean instanceof InitializingBean){
//    ((InitializingBean)bean).afterPropertiesSet();
//}

            //5.放入一级缓存
            singletonObjects.put(beanName, bean);
            earlySingletonObjects.remove(beanName);  //二级缓存是临时的需要清楚
            return bean;
        }
    }

    //获取单例池中的bean
    private Object getSingleton(String beanName) {
        if (singletonObjects.containsKey(beanName)) {
            return singletonObjects.get(beanName);
        }

        //出口
        synchronized (singletonObjects) {
            if (earlySingletonObjects.containsKey(beanName)) {
                return earlySingletonObjects.get(beanName);
            }
        }
        return null;
    }

    //ioc容器加载
    public void refersh() throws Exception {
        //1.解析配置 支持BeanDefinition
        loadBeanDefinitions();
    }

    /**
     * 根据配置信息创建BeanDefinition  底层是通过解析配置类注册beandefiniton
     */
    private void loadBeanDefinitions() {

        // 创建A BeanDefinition
        RootBeanDefinition aBeanDefinition = new RootBeanDefinition(AService.class);
        //创建B BeanDefinition
        RootBeanDefinition bBeanDefinition = new RootBeanDefinition(BService.class);

        beanDefinitionMap.put("aService", aBeanDefinition);
        beanDefinitionMap.put("bService", bBeanDefinition);
    }
}

那三级缓存用来干什么的?

三级缓存主要是处理我们涉及到需要代理的Bean的情况的。一般来说,动态代理需要Bean的初始化过程中进行创建,但是在循环依赖的这种特殊情况下,程序根本无法走到初始化这一步,所以我们需要在实例化后就进行Bean的增强。假如说我们只使用二级缓存(如下图这样写的话),对于需要进行增强的Bean会造成两个问题

1.没有遵循规范(初始化再增强

2.循环依赖多次会创建多次(A和B循环依赖,A和C循环依赖

为了解决这些问题,spring的底层引入了三级缓存(存储一个Bean工厂对象,对于需要做增强的Bean返回代理类,不需要的返回原始类)

package cn.edu.hunau;

import cn.edu.hunau.service.impl.AService;
import cn.edu.hunau.service.impl.BService;
import org.springframework.beans.factory.InitializingBean;
import org.springframework.beans.factory.ObjectFactory;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.config.BeanDefinition;
import org.springframework.beans.factory.support.RootBeanDefinition;

import java.lang.reflect.Field;
import java.util.LinkedHashMap;
import java.util.Map;
import java.util.concurrent.ConcurrentHashMap;

/**
 * @Author SuJ
 * @Date 2024 04 12 15 13.
 * 手写spring IOC底层源码来模拟spring如何利用多级缓存解决循环依赖的问题。
 **/
public class SuJApplicationContext {
    private Map<String, BeanDefinition> beanDefinitionMap = new LinkedHashMap<>();
    // 一级缓存 单例池
    private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();
    // 二级缓存 ---->  并发获取不完整bean------dcl
    private final Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>();

    //三级缓存
    private final Map<String, ObjectFactory> factoriesEarlySingletonObjects = new ConcurrentHashMap<>();


    public SuJApplicationContext() throws Exception {
        // 加载ioc容器  创建所有的bean
        refersh();

        finishBeanFactoryInitialization();
    }

    //一个个的创建bean
    private void finishBeanFactoryInitialization() {
        //循环所有的beanDefinition
        beanDefinitionMap.keySet().forEach(
                beanName -> {
                    try {
                        getBean(beanName);
                    } catch (InstantiationException e) {
                        throw new RuntimeException(e);
                    } catch (IllegalAccessException e) {
                        throw new RuntimeException(e);
                    }
                }
        );
    }

    public Object getBean(String beanName) throws InstantiationException, IllegalAccessException {
        // 1.尝试在一级缓存中获取
        Object bean = getSingleton(beanName);
        //如果存在 直接放回
        if (bean != null) {
            return bean;
        }
        synchronized (singletonObjects) {
            bean = getSingleton(beanName);
            //如果存在 直接返回
            if (bean != null) {
                return bean;
            }

            // 2.创建  ---> 实例化
            RootBeanDefinition beanDefinition = (RootBeanDefinition) beanDefinitionMap.get(beanName);
            Class<?> beanClass = beanDefinition.getBeanClass();
            Object beanNew = beanClass.newInstance();

            //1.没有遵循规范  2.循环依赖多次会创建多次
            Object beanAop = new JdkProxyBeanPostProcessor().getEarlyBeanReference(bean, beanName);

            factoriesEarlySingletonObjects.put(beanName, ()->{
                return new JdkProxyBeanPostProcessor().getEarlyBeanReference(beanNew,beanName );
            });

            // 首先将早期引用放入二级缓存
//            earlySingletonObjects.put(beanName, beanNew);

            //3. 依赖注入
            for (Field declaredField : beanClass.getDeclaredFields()) {
                //当前属性有注解
                if (declaredField.getAnnotation(Autowired.class) != null) {
                    String name = declaredField.getName();
                    Object dependBean = getBean(name);

                    declaredField.setAccessible(true);
                    declaredField.set(beanNew, dependBean);
                }
            }

            //4.初始化
//if(bean instanceof InitializingBean){
//    ((InitializingBean)bean).afterPropertiesSet();
//}

            //5.放入一级缓存
            singletonObjects.put(beanName, beanNew);
            earlySingletonObjects.remove(beanName);  //二级缓存是临时的需要清除
            factoriesEarlySingletonObjects.remove(beanName);    //三级缓存是临时的需要清除
            return beanNew;
        }
    }


    private Object getSingleton(String beanName) {
        if (singletonObjects.containsKey(beanName)) {
            return singletonObjects.get(beanName);
        }

        //出口  -- 当前是循环依赖
        synchronized (singletonObjects) {
            if (earlySingletonObjects.containsKey(beanName)) {
                return earlySingletonObjects.get(beanName);
            }

            if (factoriesEarlySingletonObjects.containsKey(beanName)) {
                ObjectFactory objectFactory = factoriesEarlySingletonObjects.get(beanName);
                // aop
                Object object = objectFactory.getObject();
                earlySingletonObjects.put(beanName, object); //解决循环依赖多次会创建多次的问题
                return object;
            }
        }
        return null;
    }

    //ioc容器加载
    public void refersh() throws Exception {
        //1.解析配置 支持BeanDefinition
        loadBeanDefinitions();
    }

    /**
     * 根据配置信息创建BeanDefinition  底层是通过解析配置类注册beandefiniton
     */
    private void loadBeanDefinitions() {

        // 创建A BeanDefinition
        RootBeanDefinition aBeanDefinition = new RootBeanDefinition(AService.class);
        //创建B BeanDefinition
        RootBeanDefinition bBeanDefinition = new RootBeanDefinition(BService.class);

        beanDefinitionMap.put("aService", aBeanDefinition);
        beanDefinitionMap.put("bService", bBeanDefinition);
    }
}

其实三级缓存的思想就是:在实例化后不是直接动态代理,而是其函数式接口放入三级缓存中,出现循环依赖时在进行调用创建代理的函数。

以上是我个人的见解,请大家多指教

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

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

相关文章

智慧园区解决方案一站式资料大全:标准规范顶层设计指南、供应商整体解决方案及售前PPT、标准白皮书、全国前50智慧园区集成商方案等全套600份,一次性打包下载

关键词&#xff1a;智慧园区解决方案&#xff0c;智慧园区整体解决方案&#xff0c;智慧园区建设总体方案设计&#xff0c;智慧园区综合管理系统&#xff0c;智慧产业园区解决方案&#xff0c;智慧产业园区规划方案&#xff0c;智慧园区建设规划方案&#xff0c;智慧工业园区建…

【数据结构1-基本概念和术语】

这里写自定义目录标题 0.数据&#xff0c;数据元素&#xff0c;数据项&#xff0c;数据对项&#xff0c;数据结构&#xff0c;逻辑结构&#xff0c;存储结构1.结构1.1逻辑结构1.2存储结构1.2.1 顺序结构1.2.2链式结构 1.3数据结构1.3.1基本数据类型1.3.2抽象数据类型1.3.2.1一个…

基于ssm幼儿资源互助共享平台的设计论文

目 录 摘 要 I Abstract II 第1章 前 言 2 1.1 研究背景 3 1.2 研究现状 3 1.3 系统开发目标 3 第2章 系统开发环境 5 2.1 SSM框架 5 2.2 JAVA简介 6 2.3 ECLIPSE 开发环境 7 2.4 Tomcat服务器 7 2.5 MySQL数据库 7 第3章 需求分析 9 3.1 需求分析 9 3.2 系统可行性分析 9 3.3…

ICV:《中美量子产业融资比较分析》

近日&#xff0c;全球前沿科技咨询公司ICV发布了A Comparative Analysis of Quantum Industry Financing in the U.S and China&#xff08;美国和中国量子产业融资比较分析&#xff09;报告。该报告旨在对中美两国在量子技术领域的投融资情况进行比较分析&#xff0c;探讨其差…

服务器数据恢复—xfs文件系统节点、目录项丢失的数据恢复案例

服务器数据恢复环境&#xff1a; EMC某型号存储&#xff0c;该存储内有一组由12块磁盘组建的raid5阵列&#xff0c;划分了两个lun。 服务器故障&#xff1a; 管理员为服务器重装操作系统后&#xff0c;发现服务器的磁盘分区发生改变&#xff0c;原来的sdc3分区丢失。由于该分区…

目标检测算法——YOLOV9——算法详解

一、主要贡献 深度网络输入数据在逐层进行特征提取和空间变换时&#xff0c;会丢失大量的信息。针对 信息丢失问题&#xff0c;研究问题如下&#xff1a; 1&#xff09;从可逆功能的角度对现有深度神经网络架构进行了理论分析&#xff0c;解释了许多过去难以解释的现象&#xf…

绝地求生:PCL大名单公布,艾伦格三巨头惨遭拆散

就在4.16号PCL官博公布了春季赛的参赛大名单&#xff0c;此次比赛不再像以前一样分为艾伦格、米拉玛和维寒迪三组&#xff0c;而是重新打乱分成了A、B、C三组。 具体名单如下 不仅多了很多新战队&#xff0c;还有一些老家伙也回到了赛场上&#xff0c;比如四大名捕的TSG。

机器学习和深度学习--李宏毅(笔记与个人理解)Day17

Day 17Convolutional Neyral Network (CNN) 卷积神经网络一般都用在image 上面比较多一些&#xff0c;所以课程的例子大多数也都是image Image Classification the same size how about for pc? 这里对于tensor 张量这个概念&#xff0c;我还是比较奇怪&#xff0c;在我认为一…

算法思想总结:链表

一、链表的常见技巧总结 二、两数相加 . - 力扣&#xff08;LeetCode&#xff09; class Solution { public:ListNode* addTwoNumbers(ListNode* l1, ListNode* l2) {//利用t来存进位信息int t0;ListNode*newheadnew ListNode(0);//创建一个哨兵节点&#xff0c;方便尾插List…

网工基础协议——TCP/UDP协议

TCP和UDP的不同点&#xff1a; TCP(Transmission Control Protocol&#xff0c;传输控制协议)&#xff1b; UDP(User Data Protocol&#xff0c;用户数据报协议)&#xff1b; TCP&#xff1a;传输控制协议&#xff0c;面向连接可靠的协议&#xff0c;只能适用于单播通信&…

【教程】一个比较良心的C++代码混淆器

这是一个比较良心的C代码混淆器&#xff0c;用于信息竞赛训练和保护代码免受抄袭。本文将介绍这个混淆器的使用方法、混淆效果和已知的一些bug。同时&#xff0c;我们也会给出一些示例来演示混淆器的具体操作。 引言 在信息竞赛训练和实际开发中&#xff0c;保护代码的安全性和…

闲不住,手写一个数据库文档生成工具

shigen坚持更新文章的博客写手&#xff0c;擅长Java、python、vue、shell等编程语言和各种应用程序、脚本的开发。记录成长&#xff0c;分享认知&#xff0c;留住感动。 个人IP&#xff1a;shigen 逛博客的时候&#xff0c;发现了一个很有意思的文章&#xff1a;数据库表结构导…

JL-32 土壤速测仪 手持便携可移动 多要素参数可选配

产品概述 土壤速测仪是一款携带方便&#xff0c;操作简单&#xff0c;集采集与存储于一体的可移动式观测仪器。由手持式速测主机、土壤类传感器、USB数据线、电源适配器、便携式手提箱等部分组成。速测仪主机可通过集线器接入不同类型的传感器&#xff0c;互不影响精度&#x…

【二分查找】Leetcode 74. 搜索二维矩阵【中等】

搜索二维矩阵 给你一个满足下述两条属性的 m x n 整数矩阵&#xff1a; 每行中的整数从左到右按非严格递增顺序排列。每行的第一个整数大于前一行的最后一个整数。 给你一个整数 target &#xff0c;如果 target 在矩阵中&#xff0c;返回 true &#xff1b;否则&#xff0c…

记录Python链接mysql数据的增删改查方法

一、添加方法 db pymysql.connect(hostlocalhost,userroot,password123456,dbpython) cursor db.cursor() sql """insert into EMPLOYEEVALUES(3,张,天爱,35,F,8000) """ try:cursor.execute(sql)db.commit() #提交后&#xff0c;数据才会变 …

Springboot+Vue项目-基于Java+MySQL的网上超市系统(附源码+演示视频+LW)

大家好&#xff01;我是程序猿老A&#xff0c;感谢您阅读本文&#xff0c;欢迎一键三连哦。 &#x1f49e;当前专栏&#xff1a;Java毕业设计 精彩专栏推荐&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb;&#x1f447;&#x1f3fb; &#x1f380; Python毕业设计 &…

Jackson 2.x 系列【28】Spring Boot 集成之 Long 精度损失

有道无术&#xff0c;术尚可求&#xff0c;有术无道&#xff0c;止于术。 本系列Jackson 版本 2.17.0 本系列Spring Boot 版本 3.2.4 源码地址&#xff1a;https://gitee.com/pearl-organization/study-jaskson-demo 文章目录 1. 问题场景2. 原因分析3. 解决方案4. 案例演示4.…

Python 物联网入门指南(七)

原文&#xff1a;zh.annas-archive.org/md5/4fe4273add75ed738e70f3d05e428b06 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 第二十四章&#xff1a;基本开关 到目前为止一定是一段史诗般的旅程&#xff01;回想一下你开始阅读这本书的时候&#xff0c;你是否曾想象…

v-for中涉及的key

一、为什么要用key&#xff1f; key可以标识列表中每个元素的唯一性&#xff0c;方便Vue高效地更新虚拟DOM&#xff1b;key主要用于dom diff算法&#xff0c;diff算法是同级比较&#xff0c;比较当前标签上的key和标签名&#xff0c;如果都一样&#xff0c;就只移动元素&#…

(十二)C++自制植物大战僵尸游戏多用户存档实现(一)

植物大战僵尸游戏开发教程专栏地址http://t.csdnimg.cn/8UFMs 游戏存档 游戏存档允许玩家保存游戏进度&#xff0c;以便在之后的时间继续游戏。通过存档&#xff0c;玩家可以暂停游戏并在需要时重新开始&#xff0c;而不必从头开始或重新完成已经完成的任务。游戏通常提供多个…