Spring缓存 & 解决循环依赖 & BeanFactory,FactoryBean区别?

简介: Spring缓存 & 解决循环依赖 & BeanFactory,FactoryBean区别?

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

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实例化完成。

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

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

@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;
}
相关文章
|
7月前
|
缓存 Java 应用服务中间件
Spring Boot配置优化:Tomcat+数据库+缓存+日志,全场景教程
本文详解Spring Boot十大核心配置优化技巧,涵盖Tomcat连接池、数据库连接池、Jackson时区、日志管理、缓存策略、异步线程池等关键配置,结合代码示例与通俗解释,助你轻松掌握高并发场景下的性能调优方法,适用于实际项目落地。
1323 5
|
监控 安全 Java
解决 Spring Boot 中 SecurityConfig 循环依赖问题的详解
本文详细解析了在 Spring Boot 中配置 `SecurityConfig` 时可能遇到的循环依赖问题。通过分析错误日志与代码,指出问题根源在于 `SecurityConfig` 类中不当的依赖注入方式。文章提供了多种解决方案:移除 `configureGlobal` 方法、定义 `DaoAuthenticationProvider` Bean、使用构造函数注入以及分离配置类等。此外,还讨论了 `@Lazy` 注解和允许循环引用的临时手段,并强调重构以避免循环依赖的重要性。通过合理设计 Bean 依赖关系,可确保应用稳定启动并提升代码可维护性。
928 0
|
7月前
|
存储 缓存 Java
Spring中@Cacheable、@CacheEvict以及其他缓存相关注解的实用介绍
缓存是提升应用性能的重要技术,Spring框架提供了丰富的缓存注解,如`@Cacheable`、`@CacheEvict`等,帮助开发者简化缓存管理。本文介绍了如何在Spring中配置缓存管理器,使用缓存注解优化数据访问,并探讨了缓存的最佳实践,以提升系统响应速度与可扩展性。
401 0
Spring中@Cacheable、@CacheEvict以及其他缓存相关注解的实用介绍
|
Java Maven 微服务
微服务——SpringBoot使用归纳——Spring Boot集成 Swagger2 展现在线接口文档——Swagger2 的 maven 依赖
在项目中使用Swagger2工具时,需导入Maven依赖。尽管官方最高版本为2.8.0,但其展示效果不够理想且稳定性欠佳。实际开发中常用2.2.2版本,因其稳定且界面友好。以下是围绕2.2.2版本的Maven依赖配置,包括`springfox-swagger2`和`springfox-swagger-ui`两个模块。
575 0
|
11月前
|
消息中间件 缓存 NoSQL
基于Spring Data Redis与RabbitMQ实现字符串缓存和计数功能(数据同步)
总的来说,借助Spring Data Redis和RabbitMQ,我们可以轻松实现字符串缓存和计数的功能。而关键的部分不过是一些"厨房的套路",一旦你掌握了这些套路,那么你就像厨师一样可以准备出一道道饕餮美食了。通过这种方式促进数据处理效率无疑将大大提高我们的生产力。
347 32
|
9月前
|
存储 缓存 NoSQL
Spring Cache缓存框架
Spring Cache是Spring体系下的标准化缓存框架,支持多种缓存(如Redis、EhCache、Caffeine),可独立或组合使用。其优势包括平滑迁移、注解与编程两种使用方式,以及高度解耦和灵活管理。通过动态代理实现缓存操作,适用于不同业务场景。
694 0
|
11月前
|
缓存 搜索推荐 CDN
HTTP缓存策略的区别和解决的问题
总的来说,HTTP缓存策略是一种权衡,需要根据具体的应用场景和需求来选择合适的策略。理解和掌握这些策略,可以帮助我们更好地优化网页性能,提高用户的浏览体验。
288 11
|
缓存 Java 应用服务中间件
微服务——SpringBoot使用归纳——Spring Boot集成Thymeleaf模板引擎——依赖导入和Thymeleaf相关配置
在Spring Boot中使用Thymeleaf模板,需引入依赖`spring-boot-starter-thymeleaf`,并在HTML页面标签中声明`xmlns:th="http://www.thymeleaf.org"`。此外,Thymeleaf默认开启页面缓存,开发时建议关闭缓存以实时查看更新效果,配置方式为`spring.thymeleaf.cache: false`。这可避免因缓存导致页面未及时刷新的问题。
485 0
|
存储 缓存 Java
Spring面试必问:手写Spring IoC 循环依赖底层源码剖析
在Spring框架中,IoC(Inversion of Control,控制反转)是一个核心概念,它允许容器管理对象的生命周期和依赖关系。然而,在实际应用中,我们可能会遇到对象间的循环依赖问题。本文将深入探讨Spring如何解决IoC中的循环依赖问题,并通过手写源码的方式,让你对其底层原理有一个全新的认识。
363 2
|
缓存 NoSQL Java
Spring Boot中的分布式缓存方案
Spring Boot提供了简便的方式来集成和使用分布式缓存。通过Redis和Memcached等缓存方案,可以显著提升应用的性能和扩展性。合理配置和优化缓存策略,可以有效避免常见的缓存问题,保证系统的稳定性和高效运行。
472 3

热门文章

最新文章

下一篇
开通oss服务