当前位置: 首页 > news >正文

Spring 循环依赖问题

Spring 的循环依赖问题是指

需要初始化的Bean A需要依赖另一个BeanB

但是BeanB 也需要依赖Bean A 导致两个Bean的初始化都无法完成

Spring是通过三级缓存机制解决循环依赖问题的

其实若是不考虑AOP 二级缓存就完全足够解决这个问题

核心思路是 提前暴露没有完全初始化的Bean、

实例化完BeanA 后就把这个Bean放入二级缓存中

这样初始化BeanB时 就可以到二级缓存中拿到这个初始化了一半的BeanA

这样BeanB就可以完成初始化 BeanA也能完成依赖注入后结束初始化

但是问题的关键是Spring的一大特性便是AOP 

若是Bean A是需要动态代理的Bean  BeanB直接把BaenA 注入肯定是不对的

需要注入代理后的对象

那我们把BeanA代理后的对象放入二级缓存可以吗?

有两方面的原因 导致不能这么做

1.动态代理注入是在Bean初始化完成之后做的,此时A尚未完成初始化,拿不到动态代理类

2.假如能提前拿到A的代理对象 那岂不是A初始化完之后又会重新代理一次? 所以不行

Spring的解决是通过三级缓存

一级缓存:存完全初始化后的对象

二级缓存:存初始化了一半的对象

三级缓存:存初始化对象所用的ObjectFactory,当一级缓存二级缓存都没有需要的Bean时就会把这个Bean的工厂放到三级缓存中

有个疑问是,为什么用ObjectFactory就可以提前获取到动态代理类的对象?又是怎么避免初始化完成后再次代理的呢?

1.ObjectFactory内会判断是否为代理对象,若是代理对象则提前生成代理对象并返回

2.会有代理对象标记,若当前Bean已经被代理过 会被标记 不会二次初始化

获取到代理对象后会放入二级缓存,若此时还有其他Baen也要获取A 则直接返回二级缓存中的对象,不会再次生成代理对象,这也是二级缓存的作用

那为什么用ObjectFactory就没有提前代理的问题呢?

http://www.dtcms.com/a/364037.html

相关文章:

  • 【代码随想录day 22】 力扣 40.组合总和II
  • 威科夫与强化学习状态
  • Spring Security 如何使用@PreAuthorize注解
  • srm信息系统数字化采购(程序代码部署程序包源码Java)
  • 实验3-传输层协议分析
  • [Java]PTA:jmu-Java-01入门-取数字浮点数
  • CentOS7安装Nginx服务——为你的网站配置https协议和自定义服务端口
  • js 获取字符串第一个字符
  • 《Visual Abstraction: A Plug-and-Play Approach for Text-Visual Retrieval》
  • 从 “容器保姆” 到 “云原生王者”:K8s 全方位指南
  • UCIE Specification详解(十三)
  • EPLAN 分散式端子:提升原理图设计效率的实用功能
  • 【C++】深入解析C++嵌套依赖类型与typename关键字
  • Jenkins Pipeline 语法
  • 【机器人概念设计软件操作手册】建筑与环境建模
  • 【服务器部署】CentOS 7/8 离线部署 Harbor v2.10.3 超详细攻略
  • docker desktop拉取镜像失败解决方案
  • ArkUI核心功能组件使用
  • pycharm无法添加本地conda解释器/命令行激活conda时出现很多无关内容
  • 【python】python进阶——pip命令
  • 单调栈与单调队列
  • 《从零搭建二叉树体系:从节点定义到子树判断的实战指南(含源码可直接运行)》
  • 利用Base64传输二进制文件并执行的方法(适合没有ssh ftp等传输工具的嵌入式离线场景)
  • TDK InvenSense CH201距离传感器
  • Photoshop用户必看:让你的PSD像JPG一样可预览
  • vim中常见操作及命令
  • 趣说IT职场30:跨团队会议话术合集:优雅反对、不留记录
  • 使用DataLoader加载本地数据
  • Elasticsearch 核心特性与应用指南
  • 【js】Promise.try VS try-catch