Spring缓存 & 解决循环依赖 & BeanFactory,FactoryBean区别?_object sharedinstance = getsingleton(beanname) 返回的-程序员宅基地

技术标签: spring  后端  javaee  

读源码的时候可以学习好的设计模式,并实践思考。

Spring有三级缓存:

一级缓存singletonObjects是线程安全的ConcurrentHashMap。

二级缓存是earlySingletonObjects,主要存放半成品的单例bean。

三级缓存singletonFactories核心是解决aop循环依赖。

第三级缓存存放原生的早期对象,二级缓存存放记过代理之后的对象。

代理分为jdk代理和cglib代理,在spring源码的方法里,postProcessBeforeInstantiation方法里,此方法可以在创建bean前返回自己的bean(可以保证单实例)。  

getSingleton()方法怎么运行的?

这里面有两个方法需要解释,isSingletonCurrentlyInCreation()判断当前单例bean是否正在创建中,也就是没有初始化完成(比如A的构造器依赖了B对象所以得先去创建B对象, 或则在A的populateBean过程中依赖了B对象,得先去创建B对象,这时的A就是处于创建中的状态。)

allowEarlyReference 是否允许从singletonFactories中通过getObject拿到对象,先从单一级缓存singletomObjects获取,没有从二级缓存earlySingletonObjects获取,没有的话再从三级缓存singletomFactory获取,当获取到值后,吧获取的实例放入二级缓存,并吧三级缓存singletomFactory里面的值移除。

beanFactory和factoryBean区别?

Spring最大的模式就是工厂模式

他们都是接口,beanFactory是帮spring维护bean的工厂。

而factoryBean是spring提供创建一些复杂的对象,比如Connnection和sqlSessionFactory对象,不能new的对象,需要我们实现factoryBean接口,这个接口里面可以判断是单实例还是多实例,主要是getObject方法来获取对象,然后把这个对象写在配置文件,或者实现配置类,通过获取工厂的getBean(id)来获取这个对象,如果想获取这个实现对象,getBean对象前面加个&来获取。

ApplicationContext是beanFactory的子接口,所以功能比beanFactory接口更强大,负责创建bean并且吧这些单实例bean保存在map中。AOP 的DI注入都是在applicationContext接口下的这些类中。ApplicationContext是留给开发者使用的aop容器接口。

Spring如何解决循环依赖?

protected  T doGetBean(final String name, @Nullable final Class requiredType,
    @Nullable final Object[] args, boolean typeCheckOnly) throws BeansException {
  
  // 尝试通过bean名称获取目标bean对象,比如这里的A对象
  Object sharedInstance = getSingleton(beanName);
  // 我们这里的目标对象都是单例的
  if (mbd.isSingleton()) {
    
    // 这里就尝试创建目标对象,第二个参数传的就是一个ObjectFactory类型的对象,这里是使用Java8的lamada
    // 表达式书写的,只要上面的getSingleton()方法返回值为空,则会调用这里的getSingleton()方法来创建
    // 目标对象
    sharedInstance = getSingleton(beanName, () -> {
      try {
        // 尝试创建目标对象
        return createBean(beanName, mbd, args);
      } catch (BeansException ex) {
        throw ex;
      }
    });
  }
  return (T) bean;
}

从spring源码开始分析:

第一步走的getSingleton先通过对象名字从缓存中获取bean,如果没有获取到,尝试获取半成品的bean,如果这里获取的是null,则进入第二个步骤。

第二步的getSingleton直接尝试创建bean对象,并且实例化之后会通过postProperlateyValue里面的populate赋值,setBeanName和setFactoryName。这里创建会调用三次doCrateBean方法,主干创建bean对象的逻辑。

这时候A对象创建的是半成品,因为依赖B对象,这时候递归调用继续走第二个getSingleton,创建B对象,这时候,B也依赖A,可以从第一个getSingleton获取到刚刚创建的半成品A。

这时候B对象实例化完成,此刻里面有一个依赖的半成品A,这时候再把递归实例化成功的B返回,此时A也依赖成功,A实例化完成。

@Nullable
protected Object getSingleton(String beanName, boolean allowEarlyReference) {
  
  // 尝试从缓存中获取成品的目标对象,如果存在,则直接返回
  Object singletonObject = this.singletonObjects.get(beanName);
  
  // 如果缓存中不存在目标对象,则判断当前对象是否已经处于创建过程中,在前面的讲解中,第一次尝试获取A对象
  // 的实例之后,就会将A对象标记为正在创建中,因而最后再尝试获取A对象的时候,这里的if判断就会为true
  if (singletonObject == null && isSingletonCurrentlyInCreation(beanName)) {
    
    synchronized (this.singletonObjects) {
      singletonObject = this.earlySingletonObjects.get(beanName);
      if (singletonObject == null && allowEarlyReference) {
        
        // 这里的singletonFactories是一个Map,其key是bean的名称,而值是一个ObjectFactory类型的
        // 对象,这里对于A和B而言,调用图其getObject()方法返回的就是A和B对象的实例,无论是否是半成品
        ObjectFactory singletonFactory = this.singletonFactories.get(beanName);
        if (singletonFactory != null) {
          
          // 获取目标对象的实例
          singletonObject = singletonFactory.getObject();
          this.earlySingletonObjects.put(beanName, singletonObject);
          this.singletonFactories.remove(beanName);
        }
      }
    }
  }
  return singletonObject;
}

spring主要就是运动递归的方式获取目标bean和其依赖的bean。

实例化bean分为两步,第一步就是实例化一个bean,第二部注入pouplate属性。首先就是递归实例化所有依赖的bean,直到某个bean没有依赖其他bean,此时就会把实例返回,然后递归将各个实例化完成的bean进行属性赋值。

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/ke1ying/article/details/121657280

智能推荐

20/0812算法题:给定一个整数数组 nums 和一个目标值 target,请你在该数组中找出和为目标值的那 两个 整数,并返回他们的数组下标。_题目描述 给定一个长度为n的数组s,还给定一个目标值m。从数组s中挑选3个数,使得这-程序员宅基地

文章浏览阅读228次。1. 两数之和给定一个整数数组nums和一个目标值target,请你在该数组中找出和为目标值的那两个整数,并返回他们的数组下标。你可以假设每种输入只会对应一个答案。但是,数组中同一个元素不能使用两遍。示例:给定 nums = [2, 7, 11, 15], target = 9因为 nums[0] + nums[1] = 2 + 7 = 9所以返回 [0, 1]解决:var twoSum = function(nums, target) { for(..._题目描述 给定一个长度为n的数组s,还给定一个目标值m。从数组s中挑选3个数,使得这

【Spring Boot采坑记】- 全局异常处理之 @ResponseStatus 和 @ExceptionHandler_@responsestatus exceptionhandler-程序员宅基地

文章浏览阅读2.1k次,点赞3次,收藏3次。Spring Boot - @ResponseStatus 和 @ExceptionHandler_@responsestatus exceptionhandler

spring cloud gateway聚合swagger_springcloud gateway 整合swagger-程序员宅基地

文章浏览阅读2.3k次,点赞3次,收藏3次。在spring cloud 的使用的时候,我发现测试起来很不方便,需要使用Postman等类似的工具来调用我们的接口,这显然是很麻烦的,那么有没有一种方式可以让我们在gateway里使用swagger来测试呢。答案是肯定的,我查阅资料发现了之前有人实现了zuul网关的聚合swagger,通过他的思路我自己写了一些类,首先需要,在gateway网关中创建三个类,下面贴出来SwaggerHandl..._springcloud gateway 整合swagger

Hive round floor ceil 用法_hive ceil-程序员宅基地

文章浏览阅读1.7w次。round 四舍五入floor 取左值ceil 取右值hive> select round(1.2356);OK1.0Time taken: 0.871 seconds, Fetched: 1 row(s)hive> select round(1.6356);OK2.0Time taken: 0.163 seconds, Fetched: 1 row(s)..._hive ceil

LTE CAT1问题记录一_4g cat.1有啥问题-程序员宅基地

文章浏览阅读1.3k次。一、硬件问题1.电源由于2G模块出现电源电平跳变,导致模块出现不工作,死机,等各种异常问题在CAT1上将基带和模块的电源分开,并且滤波电容的选取要合适,此处选择官方提供的电容并且电源供电要采取星形走线的规则。下图为走线对比。具体文章参考:电源星形走线的分析及其注意事项这篇文章说的非常详细,理解深刻如果道不同,一开始就不要为谋,不要最后一刻才来分道扬镳。这句话写得非常好。虽然在BGA的IC上才能体现出来,但是为了预防一切问题,就按照标准的星形走线方式走线2.器件布局1)首先电源器件布局_4g cat.1有啥问题

NBU 异机恢复Oracle操作步骤_nbu恢复oracle-程序员宅基地

文章浏览阅读5.8k次,点赞4次,收藏17次。一、 准备工作1. DB侧恢复服务器安装与原库相同版本的操作系统、数据库软件、NBU客户端 双向开通到NBU备份服务器的1556、13724、13720、13782、13790端口的防火墙策略(应该只要1556和13724,但为避免还原时出现其他异常,建议都开) /etc/hosts文件添加NBU备份服务器主机名及原库主机名,ping主机名测试能否解析成功 确保恢复服务器磁盘空间足够2. NBU侧/etc/hosts文件添加待恢复服务器主机名,ping主机名测试能否解析成功,若不成..._nbu恢复oracle

随便推点

openlayers 可以实现3d地图效果吗_openlayers技巧之绘制带箭头的路线-程序员宅基地

文章浏览阅读834次。这篇文章是一个技术伙伴写的,经过他授权,放到小专栏里面。效果图如下:Openlayers绘制带箭头的路线只用到了ol.FeatureStyleFunction,简单易懂,详细步骤及代码如下:第一步,创建线要素: var line_feature = new ol.Feature(); var line_geom=new ol.geom.LineString(paths); line..._openlayer 3d地图

高德地图在 vue 项目中的使用_geocoder.getaddress个人账号申请-程序员宅基地

文章浏览阅读1.1k次。高德地图在 vue 项目中的使用_geocoder.getaddress个人账号申请

Ubuntu 20.10 groovy 更换国内源_ubuntu20.10的各种国内源-程序员宅基地

文章浏览阅读9.9k次,点赞24次,收藏28次。南湖秋水夜无烟,耐可乘流直上天。且就洞庭赊月色,将船买酒白云边。—李白《游洞庭湖五首其二》Ubuntu 的软件源配置文件是 /etc/apt/sources.list。将系统自带的该文件做个备份,将该文件替换为下面内容,即可使用国内的软件源镜像。一定要备份,一定要备份,一定要备份,重要的事情说三遍。1.备份原始源文件source.listsudo cp /etc/apt/sources.list /etc/apt/sources.list.bak2.修改源文件sources.l._ubuntu20.10的各种国内源

SpringBoot下使用FreeMarker导出world,下载功能_springboot 下载freemarker-程序员宅基地

文章浏览阅读3.5k次,点赞5次,收藏7次。SpringBoot下使用FreeMarker导出world,下载功能Java中导出World文档,最早之前使用的是POI,由于比较繁琐,这次改用FreeMarker模板来进行操作,比较方便。 直奔主题吧1.准备要导出的模板文档2.将其另存为xml格式3.将该xml文件后缀改成.ftl,然后拷贝到项目中进行格式化之后就是这个样子4.编写工具类首先在添加FreeMarker的jar..._springboot 下载freemarker

ST-LINK V2.1 制作(含源码及其原理图)(type-c接口)可以配合robomaster 开发板下载口或者直接用杜邦线连接下载,支持串口调试_stlink原理图-程序员宅基地

文章浏览阅读7.8k次,点赞12次,收藏70次。标题ST-LINK V2.1 制作(含源码及其原理图)(type-c接口)可以配合robomaster 开发板下载口或者直接用杜邦线连接下载,支持串口调试基于电子爱好者,下载器是必不可少的工具,做一个自己喜欢的下载器,还是可以的,主要是便宜且简单。ST-LINK V2.1 支持SW调试 以及下载程序 ,bin文件直接拖动下载,还可以虚拟串口使用,简直不要太方便直接放图片体积只有30mmX30mm,非常小巧。支持串口和调试下载以及bin文件拖拽下载。接口也很方便,可以与robomasterA版_stlink原理图

理解serialVersionUID是什么?有什么用?如何生成?_serialversionuid是不是类似class的hashcode-程序员宅基地

文章浏览阅读335次。转自:https://www.cnblogs.com/xuxinstyle/p/11394358.html如果您曾经实现过Serializable接口,则必须遇到此警告消息The serializable class xxx does not declare a static final serialVersionUID field of type long那么......什么是serialVersionUID?serialVersionUID用作Serializable类中的版本控件。如果_serialversionuid是不是类似class的hashcode

推荐文章

热门文章

相关标签