热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

spring学习二spring三级缓存和循环依赖

1循环依赖介绍循环依赖是指两个或两个以上bean互相持有对方最终形成闭环。比如A依赖B,B依赖C,C依赖A循环依赖包括构造器依赖和属性依赖2三级缓存

1 循环依赖介绍

循环依赖是指两个或两个以上bean互相持有对方最终形成闭环。比如A依赖B,B依赖C,C依赖A


循环依赖包括构造器依赖和属性依赖


2 三级缓存解决循环依赖


2.1 spring创建Bean步骤

spring创建bean主要有3个步骤
1 createBeanInstance(实例化bean)
2 populateBean(装配bean)
3 initializeBean(初始化bean)
发生循环依赖的时候主要是在第2步


2.2 三级缓存介绍

spring管理的对象默认是单例的,那么肯定有一个地方来缓存这些对象。spring是通过三级缓存来缓存对象的

/** 一级缓存:bean name和bean实例的缓存 */
private final Map singletOnObjects= new ConcurrentHashMap<>(256);/** 三级缓存:对象工厂缓存 */
private final Map> singletOnFactories= new HashMap<>(16);/** 二级缓存:提前早期对象的缓存,还没完成初始化 */
private final Map earlySingletOnObjects= new HashMap<>(16);

实例化bean之后会先走如下代码

protected void addSingletonFactory(String beanName, ObjectFactory singletonFactory) {Assert.notNull(singletonFactory, "Singleton factory must not be null");synchronized (this.singletonObjects) {if (!this.singletonObjects.containsKey(beanName)) {this.singletonFactories.put(beanName, singletonFactory);this.earlySingletonObjects.remove(beanName);this.registeredSingletons.add(beanName);}}
}

这段代码的作用是把实例化的bean从二级缓存移除,放到三级缓存

获取bean的代码

protected Object getSingleton(String beanName, boolean allowEarlyReference) {// 从一级缓存中获取对象Object singletOnObject= this.singletonObjects.get(beanName);if (singletOnObject== null && isSingletonCurrentlyInCreation(beanName)) {// 如果一级缓存没有并且这个对象正在创建则从二级缓存获取synchronized (this.singletonObjects) {// 从二级缓存获取beansingletOnObject= this.earlySingletonObjects.get(beanName);if (singletOnObject== null && allowEarlyReference) {// 如果二级缓存也没有则从三级缓存获取ObjectFactory singletOnFactory= this.singletonFactories.get(beanName);if (singletonFactory != null) {// 如果从三级缓存获取成功则把bean从三级缓存移到二级缓存singletOnObject= singletonFactory.getObject();this.earlySingletonObjects.put(beanName, singletonObject);this.singletonFactories.remove(beanName);}}}}return singletonObject;
}

当bean初始化完全后会放入一级缓存


2.3 三级缓存解决循环依赖

假如有一个场景A引用B,B引用A,三级缓存工作原理如下
1 实例化A对象
2 把还未完全初始化的A暴露到三级缓存singletonFactories
3 装配bean
4 发现依赖B,就get B
5 B还没有被创建,则进行创建
6 B依赖A,所以get A,由于第二步已经把A放到三级缓存,所以顺利地拿到A,完成初始化,把自己放到一级缓存中
7 A顺利地拿到B,完成初始化。
三级缓存只能解决通过属性注入的循环依赖,不能解决通过构造函数注入的循环依赖。因为把bean放入三级缓存的前提是执行构造函数


2.4 三级缓存的必要性

如果只解决循环依赖问题,一级缓存足矣。但是如果仅仅使用一级缓存,那么该缓存里存放的对象既有完全初始化的,又有不完全初始化的。
如果拿到不完全初始化的对象很容易出现NPE
有人说,那好,我再建一个缓存,一级缓存用来存完全初始化的bena,二级缓存用来存不完全初始化的bean,可以了吧。这样是可以的,但是如果
一个对象被做成切面,那么该对象就会生成一个代理对象。这样依赖注入的bean仍是原始的bean,spring会抛异常。
为了解决代理对象注入的问题,加个三级缓存,里面不存具体的bean,里面存一个工厂对象。通过工厂对象,是可以拿到最终形态的代理后的bean
如果想深入了解三级缓存的必要性可以参考这位大佬的文章:spring解决循环依赖为何用三级缓存


推荐阅读
  • 深入解析Spring Cloud Ribbon负载均衡机制
    本文详细介绍了Spring Cloud中的Ribbon组件如何实现服务调用的负载均衡。通过分析其工作原理、源码结构及配置方式,帮助读者理解Ribbon在分布式系统中的重要作用。 ... [详细]
  • 本文详细介绍了Java编程语言中的核心概念和常见面试问题,包括集合类、数据结构、线程处理、Java虚拟机(JVM)、HTTP协议以及Git操作等方面的内容。通过深入分析每个主题,帮助读者更好地理解Java的关键特性和最佳实践。 ... [详细]
  • 从 .NET 转 Java 的自学之路:IO 流基础篇
    本文详细介绍了 Java 中的 IO 流,包括字节流和字符流的基本概念及其操作方式。探讨了如何处理不同类型的文件数据,并结合编码机制确保字符数据的正确读写。同时,文中还涵盖了装饰设计模式的应用,以及多种常见的 IO 操作实例。 ... [详细]
  • 实体映射最强工具类:MapStruct真香 ... [详细]
  • 本文详细介绍了Java中org.neo4j.helpers.collection.Iterators.single()方法的功能、使用场景及代码示例,帮助开发者更好地理解和应用该方法。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文详细介绍了如何构建一个高效的UI管理系统,集中处理UI页面的打开、关闭、层级管理和页面跳转等问题。通过UIManager统一管理外部切换逻辑,实现功能逻辑分散化和代码复用,支持多人协作开发。 ... [详细]
  • 2023年京东Android面试真题解析与经验分享
    本文由一位拥有6年Android开发经验的工程师撰写,详细解析了京东面试中常见的技术问题。涵盖引用传递、Handler机制、ListView优化、多线程控制及ANR处理等核心知识点。 ... [详细]
  • 深入解析 Apache Shiro 安全框架架构
    本文详细介绍了 Apache Shiro,一个强大且灵活的开源安全框架。Shiro 专注于简化身份验证、授权、会话管理和加密等复杂的安全操作,使开发者能够更轻松地保护应用程序。其核心目标是提供易于使用和理解的API,同时确保高度的安全性和灵活性。 ... [详细]
  • 本文深入探讨了 Java 中的 Serializable 接口,解释了其实现机制、用途及注意事项,帮助开发者更好地理解和使用序列化功能。 ... [详细]
  • XNA 3.0 游戏编程:从 XML 文件加载数据
    本文介绍如何在 XNA 3.0 游戏项目中从 XML 文件加载数据。我们将探讨如何将 XML 数据序列化为二进制文件,并通过内容管道加载到游戏中。此外,还会涉及自定义类型读取器和写入器的实现。 ... [详细]
  • 在 Swift 编程中,遇到错误提示“一元运算符 '!' 不能应用于 '()' 类型的操作数”,通常是因为尝试对没有返回值的方法或函数应用逻辑非运算符。本文将详细解释该错误的原因,并提供解决方案。 ... [详细]
  • 本文介绍如何使用阿里云的fastjson库解析包含时间戳、IP地址和参数等信息的JSON格式文本,并进行数据处理和保存。 ... [详细]
  • 本文详细介绍了 Java 中 org.apache.xmlbeans.SchemaType 类的 getBaseEnumType() 方法,提供了多个代码示例,并解释了其在不同场景下的使用方法。 ... [详细]
author-avatar
天堂调度长
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有