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

泰州企业模板建站揭阳网站建设团队

泰州企业模板建站,揭阳网站建设团队,开源企业网站,社交网站源代码1. Spring使用三级缓存来解决循环依赖问题 Spring使用三级缓存来解决循环依赖问题#xff0c;‌而不是使用两级缓存。‌ 在Spring框架中#xff0c;‌解决循环依赖的关键在于正确地管理Bean的生命周期和依赖关系。‌循环依赖指的是两个或多个Bean相互依赖#xff0c;‌如果…1.  Spring使用三级缓存来解决循环依赖问题 Spring使用三级缓存来解决循环依赖问题‌而不是使用两级缓存。‌ 在Spring框架中‌解决循环依赖的关键在于正确地管理Bean的生命周期和依赖关系。‌循环依赖指的是两个或多个Bean相互依赖‌如果处理不当‌可能会导致Bean无法正确初始化。‌Spring通过引入三级缓存机制来解决这个问题‌而不是使用两级缓存。‌ 1.  一级缓存‌存储已经初始化完成的Bean实例。‌当其他Bean依赖这个Bean时‌可以直接从缓存中获取实例‌避免重复创建。‌ 2.  二级缓存‌在某些情况下‌二级缓存用于存储Bean的代理对象或原始实例‌以避免重复创建。‌然而‌在处理循环依赖时‌二级缓存不足以解决问题‌因为它不能确保在需要时总是提供正确的对象实例。‌ 3.  三级缓存‌这是为了解决二级缓存的局限性而引入的。‌三级缓存存储了一个函数接口及其实现‌用于创建动态代理对象或调用BeanPostProcessor。‌通过这种方式‌Spring可以确保在需要时提供正确的对象实例‌无论是原始对象还是其代理对象‌从而解决了循环依赖的问题。‌ 通过这种方式‌Spring能够确保在复杂的依赖关系中正确地管理和初始化Bean‌避免了因循环依赖导致的初始化失败问题。‌这种三级缓存机制不仅提高了代码的可维护性‌还优化了性能‌确保了Spring应用程序的稳定性和可靠性 singletonObjects 一级缓存存储单例对象Bean 已经实例化初始化完成。 earlySingletonObjects 二级缓存存储 singletonObject这个 Bean 实例化了还没有初始化。 singletonFactories 三级缓存存储 singletonFactory。 2.  Bean 的创建过程 Service public class CircularServiceA {private String fieldA 字段 A; }通过上面的流程可以看出 Spring 在创建 Bean 的过程中重点是在 AbstractAutowireCapableBeanFactory 中的以下三个步骤 实例化 createBeanInstance 其中实例化 Bean 并对 Bean 进行赋值像例子中的 fieldA 字段在这里就会赋值。 属性注入 populateBean 可以理解为对 Bean 里面的属性进行赋值。(会依赖其他 Bean) 初始化 initializeBean 执行初始化和 Bean 的后置处理器。 实例化赋值源码可以阅读 BeanUtils.instantiateClass(constructorToUse) 3.  如果要依赖其他 Bean 呢 那如果 CircularServiceA 依赖了其他 Bean 呢 Service public class CircularServiceA {private String fieldA 字段 A;Autowiredprivate CircularServiceB circularServiceB;} Service public class CircularServiceB {}当 A 依赖了 B 的时候在 createBeanInstance 这一步并不会对 B 进行属性赋值。 而是在 populatedBean 这里查找依赖项并创建 B。 4.  循环依赖下的创建过程 循环依赖的场景在上一篇文章已经有所讲解这里仅仅画图说明一下。 Service public class CircularServiceA {private String fieldA 字段 A;Autowiredprivate CircularServiceB circularServiceB;} Service public class CircularServiceB {Autowiredprivate CircularServiceA circularServiceA; }在 A 和 B 循环依赖的场景中 B populatedBean 查找依赖项 A 的时候从一级缓存中虽然未获取到 A但是发现 A 在创建中。 此时从三级缓存中获取 A 的 singletonFactory 调用工厂方法创建 getEarlyBeanReference A 的早期引用并返回。 B 引用到 A B 就可以初始化完毕然后 A 同样也可以初始化完毕了。 5.  二级缓存能否解决循环依赖 通过上面的图仔细分析一下其实把二级缓存拿掉在 B 尝试获取 A 的时候直接返回 A 的实例是不是也是可以的 答案是可以的 但是为什么还是用三级缓存呢 网上的很多资料说是和动态代理有关系那就从动态代理的方面继续往下分析分析。 6.  动态代理的场景Bean 的创建过程 在 JavaConfig配置类 上添加 EnableAspectJAutoProxy 注解开启 AOP 通过 Debug 循序渐进看一看动态代理对循环依赖的影响。 动态代理下Bean 的创建过程 Service public class CircularServiceA {private String fieldA 字段 A;public void methodA() { System.out.println(方法 A 执行);} } Aspect Component public class AspectA {Before(execution(public void com.liuzhihang.circular.CircularServiceA.methodA()))public void beforeA() {System.out.println(beforeA 执行);} } 只有 A 的情况下给 A 添加切面开始 Debug。 前面的流程都相同在 initializeBean 开始出现差异。 这一步需要初始化 Bean 并执行 Bean 的后置处理器。 其中有一个处理器为 AnnotationAwareAspectJAutoProxyCreator 其实就是加的注解切面会跳转到 AbstractAutoProxyCreator 类的 postProcessAfterInitialization 方法 如图所示wrapIfNecessary 方法会判断是否满足代理条件是的话返回一个代理对象否则返回当前 Bean。 后续调用 getProxy 、createAopProxy 等等最终执行到下面一部分。 最终会执行到这里AOP 代理相关的就不细看了。 一路放行直到 initializeBean 执行结束。 此时发现A 被替换为了代理对象。 所以 doCreateBean 返回以及后面放到一级缓存中的都是代理对象。 7.有循环依赖的动态代理 这一次把循环依赖打开 Service public class CircularServiceA {private String fieldA 字段 A;Autowiredprivate CircularServiceB circularServiceB;public void methodA() { System.out.println(方法 A 执行);} } Aspect Component public class AspectA {Before(execution(public void com.liuzhihang.circular.CircularServiceA.methodA()))public void beforeA() { System.out.println(beforeA 执行);}} Service public class CircularServiceB {Autowiredprivate CircularServiceA circularServiceA;public void methodB() {} } Aspect Component public class AspectB {Before(execution(public void com.liuzhihang.circular.CircularServiceB.methodB()))public void beforeB() { System.out.println(beforeB 执行);}}开始 Debug前面的一些列流程都和正常的没有什么区别。而唯一的区别在于创建 B 的时候需要从三级缓存获取 A。 此时在 getSingleton 方法中会调用singletonObject singletonFactory.getObject(); 有时会比较疑惑 singletonFactory.getObject() 调用的是哪里 所以这一块调用的是 getEarlyBeanReference开始遍历执行 BeanPostProcessor。 看到 wrapIfNecessary 就明白了吧这块会获取一个代理对象。 也就是说此时返回并放到二级缓存的是一个 A 的代理对象。 这样 B 就创建完毕了 到 A 开始初始化并执行后置处理器了因为 A 也有代理所以 A 也会执行到 postProcessAfterInitialization 这一部分 但是在执行 wrapIfNecessary 之前会先判断代理对象缓存是否有 A 了。 this.earlyProxyReferences.remove(cacheKey) ! bean 但是这块获取到的是 A 的代理对象。肯定是 false 。 所以不会再生成一次 A 的代理对象。 8.总结 可以看到循环依赖下有没有代理情况下的区别就在 singletonObject singletonFactory.getObject(); 在循环依赖发生的情况下 B 中的 A 赋值时 无代理getObject 直接返回原来的 Bean 有代理getObject 返回的是代理对象 然后都放到二级缓存。 9.为什么要三级缓存? 假设去掉三级缓存 去掉三级缓存之后Bean 直接创建 earlySingletonObjects 看着好像也可以。 如果有代理的时候在 earlySingletonObjects 直接放代理对象就行了。 但是会导致一个问题在实例化阶段就得执行后置处理器判断有 AnnotationAwareAspectJAutoProxyCreator 并创建代理对象。 这么一想是不是会对 Bean 的生命周期有影响。 同样先创建 singletonFactory 的好处就是在真正需要实例化的时候再使用 singletonFactory.getObject() 获取 Bean 或者 Bean 的代理。相当于是延迟实例化。 假设去掉二级缓存 如果去掉了二级缓存则需要直接在 singletonFactory.getObject() 阶段初始化完毕并放到一级缓存中。 那有这么一种场景B 和 C 都依赖了 A。 要知道在有代理的情况下 singletonFactory.getObject() 获取的是代理对象。 而多次调用 singletonFactory.getObject() 返回的代理对象是不同的就会导致 B 和 C 依赖了不同的 A。 那如果获取 B 到之后直接放到一级缓存然后 C 再获取呢 一级缓存放的是已经初始化完毕的 Bean要知道 A 依赖了 B 和 C A 这时候还没有初始化完毕。
http://www.zqtcl.cn/news/879051/

相关文章:

  • 中山网站制作服务公司做环评的网站
  • 江山市住房和城乡建设局网站iis部署网站 错误400
  • 网站域名如何备案建设厅公积金中心网站
  • 网站怎么建设?电子商务网站开发相关技术
  • 苏州网站设计公司济南兴田德润厉害吗python基础教程第3版
  • 网站多久备案一次电子商务平台信息系统建设
  • 网站开发方面的文献自己怎么建个免费网站吗
  • 建设网站前的市场分析百度竞价推广是什么
  • 专门做照片书的网站阳谷聊城网站优化
  • 国际贸易相关网站网站建设的目标与思路
  • 小型网站建设费用云南网站建设企业推荐
  • 设备租赁业务网站如何做看板娘 wordpress
  • 上海网站设计工作室二手交易网站建设目标
  • 深圳智能响应网站建设平面设计基础教程
  • 网站建设 推广全流程案例分析网站
  • 企业建网站多少钱怎样做网站挣钱
  • 经营性质的网站asp.ne做网站
  • 天津都有哪些制作网站开网站挣不挣钱
  • 网站建设云技术公司推荐重庆网页设计培训
  • 做房产网站不备案可以吗北京爱空间装修公司
  • 手机网站是用什么开发的厦门公司网站制作流程
  • 网站是广西住房和城乡建设厅wordpress插件数据库存在哪
  • 网站图片如何做链接网站制作及管理教程
  • 企业建立企业网站有哪些优势?app下载排行榜
  • 广州天河网站建设gif在线制作
  • 建个大型网站要多少钱小程序开发公司简介
  • 定制建设网站商洛做网站的公司
  • 网站建设目标活动策划书模板
  • 手机网站空间申请做网站需要空间跟域名吗
  • 洛阳专业网站设计开发制作建站公司长沙网站制作app开发公司