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

Spring解决循环依赖为什么需要三级缓存?

前言

什么是循环依赖呢?我们抛开Spring这个框架来聊下什么是循环依赖,循环依赖可能在我们平时的开发过程中是属于比较常见的。Spring容器最大的功能就是对bean的生命周期进行管理,每个bean在创建的过程中,需要得到一个完整的bean需要对bean的所有属性进行赋值,如果两个bean出现了相互依赖的情况,如果Spring没有处理循环依赖,那么出现的结果就是在bean的创建过程中出现相互依赖,导致这个bean永远无法创建出来,则就导致一直在相互创建,那么Spring是如何来解决循环依赖的呢?

什么情况下会循环依赖

1.先看如下demo: B和A相互循环依赖

@Component
public class B {@Autowiredprivate A a;
}@Component
public class A {@Autowiredprivate B b;
}

启动项目:结果没有报错。

2.加入异步逻辑修改

@Component
public class A {@Autowiredprivate B b;@Asyncpublic void test(){}
}@Component
public class B {@Autowiredprivate A a;
}
@EnableAsync
public class App {public static void main(String[] args) {ApplicationContext ctx = new AnnotationConfigApplicationContext(App.class);}}

启动后 :

解决方案:加入lazy注解

@Component
public class B {@Autowired@Lazyprivate A a;
}
@Component
public class A {@Autowiredprivate B b;@Asyncpublic void test(){}
}

启动后:没有异常

上面发现使用@Async异步注解,循环依赖就会报错,有可能是因为有了@Async注解修饰的方法,其对应的类被代理了,那代理了就会报错么?我们继续尝试事务注解看看。

@Component
public class A {@Autowiredprivate B b;@Transactionalpublic void test(){}
}@Component
public class B {@Autowiredprivate A a;
}
@EnableTransactionManagement
public class App {public static void main(String[] args) {ApplicationContext ctx = new AnnotationConfigApplicationContext(App.class);}}

启动后:正常,没有报错。

于是我们不经要问:

  1. 循环依赖本来不会报错,为何添加@Async异步注解后就会导致报错

  1. 为何添加@Transactional注解就不会报错

  1. 使用了@Async异步注解的循环依赖,为何可以使用@lazy注解解决

我们要想清楚上面的问题,就需要了解Bean的生命周期。

Bean的生命周期

一个简单的Bean生命周期如下:

问题出现就属性赋值这里:

由图:我们知道,当B也依赖A时,需要去容器中找到A,A已经实例化了,只是还没属性赋值,所以,不应该再实例化,解决方案:在A创建的实例化后,用一个map存起来A来不就行了么?于是有了二级缓存

似乎上面已经可以解决循环依赖了,但细想一下我们会发现问题:

通过上面的逻辑,我们发现了问题所在,B赋值属性A时,如果从Map中直接获取,那么得到的是原生对象,如果后续A没有被代理,一切没问题,如果A被代理了,那么B得到的对象就不对了,怎么解决,如果我们将aop提前是不是解决了问题。

由于A对象的Aop方式提前了,那么B依赖的A就是代理对象了,A对象执行赋值后,后续到Aop这一步,会判断是否已经AOP过了,是的话就不会再Aop了,问题来了:如果C也跟A相互依赖,难道C去依赖A时,也要通过ObjectFactory获取A的代理对象么?如果是这样,A就存在2个代理对象了,A是单例的,因此这样不行,于是产生了一个新的缓存,我们称之为三级缓存。

于是,spring似乎完美解决了循环依赖问题?但为何使用@Async进行异步代理,会报错?

我们看看报错的原因就知道:

那为何@Transactional修饰就没问题呢?

原因是因为:ObjectFactory.getObject()方法可以产生代理对象

为何使用@lazy注解修饰就能解决问题呢?

我们看看源码:

从源码来看,为何@Aync注解修饰,不能在ObjectFactory.getObject()方法实现代理对象:

而@Tranctional注解相关的处理器

那么问题?如果A已经在getObject()方法后产生了代理类,后续init()方法后,还会执行代理么?答案是不会了,因为:

总结

1、三级缓存各自的作用

第一级缓存存的是对外暴露的对象,也就是我们应用需要用到的

第二级缓存的作用是为了处理循环依赖的对象创建问题,里面存的是半成品对象或半成品对象的代理对象

第三级缓存的作用处理存在 AOP + 循环依赖的对象创建问题,能将代理对象提前创建

2、Spring 为什么要引入第三级缓存

严格来讲,第三级缓存并非缺它不可,因为可以提前创建代理对象

提前创建代理对象只是会节省那么一丢丢内存空间,并不会带来性能上的提升,但是会破环 Spring 的设计原则

Spring 的设计原则是尽可能保证普通对象创建完成之后,再生成其 AOP 代理(尽可能延迟代理对象的生成)

所以 Spring 用了第三级缓存,既维持了设计原则,又处理了循环依赖;牺牲那么一丢丢内存空间是愿意接受的

http://www.lryc.cn/news/17778.html

相关文章:

  • Android源码分析 - 回顾Activity启动流程
  • PDMS二次开发(一)——PML类型程序类型与概念
  • 一文揭晓:手机号码归属地api的作用是什么?
  • 电容的结构分类介质封装及应用场景总结
  • 数据结构初阶——时间复杂度与空间复杂度
  • 深度学习之“制作自定义数据”--torch.utils.data.DataLoader重写构造方法。
  • #G. 求约数个数之六
  • 如何为Java文件代码签名及添加时间戳?
  • Xamarin.Forsm for Android 显示 PDF
  • RK3399平台开发系列讲解(LED子系统篇)LED子系统详解
  • LeetCode 432. 全 O(1) 的数据结构
  • 再析jvm
  • 社招前端二面面试题总结
  • 人人能读懂redux原理剖析
  • uniCloud云开发----7、uniapp通过uni-swiper-dot实现轮播图
  • IM即时通讯构建企业协同生态链
  • Python实现构建gan模型, 输入一个矩阵和两个参数值,输出一个矩阵
  • 开学准备哪些电容笔?ipad触控笔推荐平价
  • 放下和拿起 解放自己
  • 100%BIM学员的疑惑:不会CAD可以学Revit吗?
  • 经常会采坑的javascript原型应试题
  • 完全背包—动态规划
  • 消息队列MQ介绍
  • C语言进阶(八)—— 链表
  • 手工测试用例就是自动化测试脚本——使用ruby 1.9新特性进行自动化脚本的编写
  • RockerMQ简介和单节点部署
  • SFP光纤笼子 别称 作用 性能要点 工程要素
  • [HarekazeCTF2019]Easy Notes
  • Java学习-IO流-字符流-FileReader
  • python攻陷米哈游《元神》数据?详情请看文章。。