map 循环_被问到Spring循环依赖怎么解决?秀给面试官看!内附图解

daefad81b7e5fbfd0649824d089d1b3d.png

不知道最近有没有被一道Java面试题刷爆朋友圈,Spring框架的循环依赖如何解决。我收到了不少粉丝的提问,在了解到之后,也去网上查询了一些资料,自己也询问了身边的同事,总结出以下几个方面,今天就和我来看一看吧~

7b6eb659d8ecb40b57b3c920da401459.png

寻常情况下,如果问Spring内部怎么去解决循环的依赖性,一定是单默认的单例Bean中,属性互相引用的场景。假设几个Bean之间的互相引用,甚至循环依赖自己。

44b9b3922f505c30fb5732efc06e2249.png

ce5658222b21bf29761d4ec3bec0e471.png

根据上面的两个图,我们先说一下循环依赖与原型的场景是不互相支持的,通常会走到AbstractBeanFactory类中下面的判断,然后反馈回异常问题。

if (isPrototypeCurrentlyInCreation(beanName)) {  throw new BeanCurrentlyInCreationException(beanName); } 

原因其实并不难,如果要创建一个新的A就会发现需要注入原型字段B,当创建新的原型字段B时又发现需要新的A。这就很尴尬了,禁止套娃!总不能靠猜去判断先是StackOverflow还是OutOfMemory?这也太难了吧~

所以Spring怕你猜起来困难,就非常贴心的出现了BeanCurrentlyInCreationException。真不愧是我最爱的框架。

在基于在构造器上的循环依赖,这就不必再多说了,官方文档有很明显的指示,想让构造器注入去支持循环依赖?这就不可能了,改代码吧······

那么默认单例的属性注入场景,那么Spring对循环依赖是如何支持的呢?

Spring解决循环依赖

这时候我们就不得不说到Spring的内部了,它内部维护了三个Map,这是什么?就是我们常说的三个缓存级别。这是为了让更好理解,其实并没有官方名字坐实这个三级缓存的概念。不过这不重要,接着看就是了。

在Spring的DefaultSingletonBeanRegistry类中,你就会发现它的上面有三个Map:

1.singletonObjects。这个或许是我们最熟悉的部分了,我们通常叫它:单例池,容器,它其实就是缓存创建完成单例Bean的地方。

2.singletonFactories。用来映射创建Bean的原始工厂。

3.earlySingletonObjects。它用来映射Bean的早期引用,这意思就是Map里的Bean并不完整,与其称之为Bean,倒不说它只是一个Instance.

再往后的两个Map就更像是一个“垫脚石”了,创建Bean时用了一下,用完就清理了。

循环依赖的本质

了解本质之后才能知道如何解决,刚才说了Spring如何处理循环依赖,首先,我们跳出“阅读源码”的思维,举个例子,如果让你实现下面的功能,你会如何去做?

1.将指定的一些类实例为单例

2.类中的字段同样实例为单例

3.必须支持循环依赖

假设类A是存在的,那么

public class A {  private B b;  
}  // 类B:  
public class B {  private A a;  
} 

看到了吗?其实就是让你模仿一下Spring,假设A和B被修饰,而且类之间的字段假设是通过Autowired修饰,然后放到Map里面,经过处理之后再放到Map里面。

b2b3ba3b5bc0238e505eb6639e40cf2d.png

其实上述并不是“Spring如何去解决循环依赖”而是循环依赖的基本本质,其实在网上可以搜索到很多例子,完全可以去百度一下看一看,这可以让你不在阅读的泥潭里陷得太深进而忽略了问题本质,如果实在是看不懂,逆推Spring的实现原因效果会好很多。

问题的本质竟然在于two sum?

说到这里有没有觉得似曾相识?好像在什么时候见过似的,没错,和two sum的解题是很相似的。什么?你不知道two sum?two sum是刷题网站leetcode序号为1的题,也就是大多人的算法入门的第一题。经常有梗对于这个two sum,感兴趣的可以去看看。咳咳,跑题了,我们再回来

问题的内容是:先给你规定数组,再给定一个数字。再返回到数组里面允许通过相加得到指定数字的两个索引。我们举个例子,给定nums = [2, 7, 11, 15], target = 9 那么要返回 [0, 1],因为2 + 7 = 9这道题的优解是,一次遍历+HashMap:

class Solution {  public int[] twoSum(int[] nums, int target) {  Map<Integer, Integer> map = new HashMap<>();  for (int i = 0; i < nums.length; i++) { int complement = target - nums[i];  if (map.containsKey(complement)) {  return new int[] { map.get(complement), i };  }              map.put(nums[i], i);  }          throw new IllegalArgumentException("No two sum solution");  }  } 

这个时候就需要先去Map中寻找我们需要的数字,如果没有,那么就将数字先保存到Map里面,再寻找到需要的数字时,一起返回即可。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

调用ThunderAgent 迅雷局域网版的开发

P2P方式的下载&#xff0c;为我们获取互联网资料提供了一个强有力的工具&#xff0c;它可以争取最大的带宽&#xff0c;获取最大的下载速度。最早采用P2P方式下载的工具有BT、电骡等&#xff0c;后来许多传统“多线程断点续传类”软件&#xff0c;也增加了对的支持&#xff0c;…

详解虚拟机中为Linux添加硬盘

Linux添加硬盘是在原来安装的硬盘空间不够或者需要使用其他硬盘上的东西时候的解决办法&#xff0c;因为大多数初学者习惯使用虚拟机&#xff0c;这里以在Vmware虚拟机中实现Linux添加硬盘的具体步骤来详细介绍说这个问题。 Vmware中为linux增加硬盘 1. 在vmware的setting中为虚…

vue 常用功能和命令

1. vue-cli 构建项目 # 全局安装 vue-cli $ npm install --global vue-clif # 创建一个基于 webpack 模板的新项目 $ vue init webpack your-project-name # 安装依赖 $ npm install # 进入项目 $ cd your-project-name # 开发版本打包并运行 $ npm run dev# 线上环境整个项目打…

项目管理最佳实践方法_项目管理:控制项目进度最佳实践

5分钟站立会议5 MinutesStand-up Meeting&#xff08;5分钟站立会议&#xff09;是实践中项目进度管理的好办法。5分钟站立会议时&#xff0c;项目团队成员在固定时间&#xff08;如每天上午8:30&#xff5e;8:35&#xff09;、固定地点&#xff0c;每天站着围在一起&#xff0…

《海龟交易法则》经典梳理

序言&#xff0c;第一&#xff5e;第三章 海归交易法则序言本书是历史上最好的5本交易学著作之一的理由&#xff1a;1&#xff0c;重要的不是交易系统而是交易者贯彻交易系统的能力。2&#xff0c;用浅显易懂的语言阐述行为金融学的某些原理如何用于交易和影响交易。3&#xff…

Unix下C程序内存泄漏检测工具Valgrind安装与使用

Valgrind是一款用于内存调试、内存泄漏检测以及性能分析的软件开发工具。 Valgrind的最初作者是Julian Seward&#xff0c;他于2006年由于在开发Valgrind上的工作获得了第二届Google-OReilly开源代码奖。 Valgrind遵守GNU通用公共许可证条款&#xff0c;是一款自由软件。 官网…

hibernate保存失败_Hibernate:保存与保存并保存或更新

hibernate保存失败save和saveOrUpdate之间的区别是什么或save和persist之间的区别是任何Hibernate面试中常见的面试问题&#xff0c;就像Hibernate中get和load方法之间的区别一样。 Hibernate Session类提供了几种通过save &#xff0c; saveOrUpdate和persist等方法将对象保存…

BZOJ 2957 楼房重建-线段树

这个题最主要的是解决一个统计答案的问题。 首先我们注意到&#xff0c;只要考虑右区间的答案统计就好了。 记左区间的最大值为K&#xff0c;当前右区间为P。 我们把当前右区间又分成两个子区间&#xff0c;s1,s2。 那么如果s1的最大值比K小&#xff0c;那么显然只要递归处理s2…

matlab 移动平均_两所高校被禁用MATLAB背后,是工业设计能力之争

不管是MATLAB&#xff0c;还是EDA&#xff0c;都指向了中国核心工业软件缺失的问题。 作者&#xff5c; Decode 邮箱&#xff5c;oudibjoutlook.com来源丨人民数字与品玩联合出品一款工科软件&#xff0c;最近成为科研界议论的焦点。2020 年 6 月 6 日开始&#xff0c;哈尔滨工…

MySQL 当记录不存在时insert,当记录存在时update

MySQL 当记录不存在时insert&#xff0c;当记录存在时更新 网上基本有三种解决方法。 第一种&#xff1a; 示例一&#xff1a;insert多条记录 假设有一个主键为 client_id 的 clients 表&#xff0c;可以使用下面的语句&#xff1a; INSERT INTO clients (client_id, client_na…

如何使用Valgrind memcheck工具进行C/C++的内存泄漏检测

英文原文&#xff1a;How to Detect Memory Leaks Using Valgrind memcheck Tool for C / C 系统编程中一个重要的方面就是有效地处理与内存相关的问题。你的工作越接近系统&#xff0c;你就需要面对越多的内存问题。有时这些问题非常琐碎&#xff0c;而更多时候它会演变成一个…

如何在Activiti中使用瞬态变量

我们昨天发布的Activiti v6 Beta3中已经加入了一个非常需要的功能-临时变量。 在本文中&#xff0c;我将向您展示一个示例&#xff0c;该示例说明如何使用瞬态变量来覆盖一些以前不可能&#xff08;或最佳&#xff09;的高级用例。 到目前为止&#xff0c;Activiti中的所有变量…

python羊车门问题_「羊车门」经典概率题中不换门选中车的概率是多少?

今天用Python求解「羊车门」经典的概率问题,对概率学基础和Python语法的灵活运用有所收货.本次「羊车门」求解过程采用的是:穷举法计算概率已验证概率学基础理论.期间重点借鉴了奥卡姆剃刀的博客和 南葱&#xff1a;「羊车门」经典概率题中不换门选中车的概率是多少&#xff1f…

非标协议外设LCD1602

概述 LCD1602 &#xff08; Liquid Crystal Display &#xff09;是一种工业字符型液晶&#xff0c;能够同时显示 1602 即 32 字符 (16 列两行) 引脚说明 第 1 脚 : VSS 为电源地 第 2 脚 : VDD 接 5V 正电源 第 3 脚 : VL 为液晶显示器对比度调整端 , 接正电源…

50: Luogu P4568 分层图

分层图最短路模板 #include <iostream> #include <cstdio> #include <cstdlib> #include <ctime> #include <queue> #include <cstring>using namespace std;const int M 2e6 5e5 10;#define gc getchar() inline int read() {int x 0…

C++编程笔记:dll的生成与使用

1.动态链接库&#xff08;dll&#xff09;概述 没接触dll之前觉得它很神秘&#xff0c;就像是一个黑盒子&#xff0c;既不能直接运行&#xff0c;也不能接收消息。它们是一些独立的文件&#xff0c;其中包含能被可执行程序或其他dll调用来完成某项工作的函数&#xff0c;只有在…

如何通过IP地址分辨公网、私网、内网、外网

如何通过IP地址分辨公网、私网、内网、外网内、外网是相对于防火墙而言的&#xff0c;在防火墙内部叫做内网&#xff0c;反之就是外网。在一定程度上外网等同于公网&#xff0c;内网等同于私网。地址为如下3个区域就是处于私网&#xff1a;1&#xff1a;10.*.*.*2&#xff1a;1…

python画动态表情包_真香!一行Python代码,帮你制作小姐姐的表情包,靠谱吗?...

原标题&#xff1a;真香&#xff01;一行Python代码&#xff0c;帮你制作小姐姐的表情包&#xff0c;靠谱吗&#xff1f;(我的IU女神)对于小姐姐的动态表情包&#xff0c;相必我们大多数人都不会拒绝&#xff0c;而且都会选择默默的将其收藏(不要问我怎么知道的)&#xff0c;一…

mongodb分片

mongodb分片&#xff1a; 本次是用三台主机搭建3个集群&#xff08;主、备、仲裁&#xff09;作为三个分片&#xff0c;一个集群&#xff08;主、备、备&#xff09;做为config服务器&#xff0c;三个mongos单点做路由&#xff0c;每台5个&#xff0c;一共15个。 新建一个mongo…

批量添加PDF帐号目录

本文参考&#xff1a;https://blog.csdn.net/qq_34104395/article/details/78766400然后根据需要整理的。如作者介意请留言&#xff0c;本人会尽快处理&#xff01; 准备材料&#xff1a; 下载工具FreePic2Pdf&#xff08;在本博客上传资料上找PDF转换工具包&#xff09; 找到…