网上关于Spring循环依赖的博客太多了,有很多都分析的很深入,写的很用心,甚至还画了时序图、流程图帮助读者理解,我看了后,感觉自己是懂了,但是闭上眼睛,总觉得还没有完全理解,总觉得还有一两个坎过不去,对我这种有点笨的人来说,真的好难。当时,我就在想,如果哪一天,我理解了Spring循环依赖,一定要用自己的方式写篇博客,帮助大家更好的理解,等我理解后,一直在构思,到底怎么应该写,才能更通俗易懂,就在前几天,我想通了,这么写应该更通俗易懂。在写本篇博客之前,我翻阅了好多关于Spring循环依赖的博客,网上应该还没有像我这样讲解的,现在就让我们开始把。

什么是循环依赖

一言以蔽之:两者相互依赖。

在开发中,可能经常出现这种情况,只是我们平时并没有注意到原来我们写的两个类、甚至多个类相互依赖了,为什么注意不到呢?当然是因为没有报错,而且一点问题都木有,如果报错了,或者产生了问题,我们还会注意不到吗?这一切都是Spring的功劳,它在后面默默的为我们解决了循环依赖的问题。

如下所示:

@Configuration
@ComponentScan
public class AppConfig {
}
@Service
public class AuthorService {@AutowiredBookService bookService;
}
@Service
public class BookService {@AutowiredAuthorService authorService;
}
public class Main {public static void main(String[] args) {ApplicationContext annotationConfigApplicationContext = new AnnotationConfigApplicationContext(AppConfig.class);BookService bookService = (BookService) annotationConfigApplicationContext.getBean("bookService");System.out.println(bookService.authorService);AuthorService authorService = (AuthorService) annotationConfigApplicationContext.getBean("authorService");System.out.println(authorService.bookService);}
}

运行结果:

com.codebear.springcycle.AuthorService@63376bed
com.codebear.springcycle.BookService@4145bad8

可以看到BookService中需要AuthorService,AuthorService中需要BookService,类似于这样的就叫循环依赖,但是神奇的是竟然一点问题没有。

当然有些小伙伴可能get不到它的神奇之处,至于它的神奇之处在哪里,我们放到后面再说。

任何循环依赖,Spring都能解决吗

不行。

如果是原型 bean的循环依赖,Spring无法解决:

@Service
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class BookService {@AutowiredAuthorService authorService;
}
@Service
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class AuthorService {@AutowiredBookService bookService;
}

启动后,令人恐惧的红色字体在控制台出现了:

image.png

如果是构造参数注入的循环依赖,Spring无法解决:

@Service
public class AuthorService {BookService bookService;public AuthorService(BookService bookService) {this.bookService = bookService;}
}
@Service
public class BookService {AuthorService authorService;public BookService(AuthorService authorService) {this.authorService = authorService;}
}

还是讨厌的红色字体:

image.png

循环依赖可以关闭吗

可以,Spring提供了这个功能,我们需要这么写:

public class Main {public static void main(String[] args) {AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();applicationContext.setAllowCircularReferences(false);applicationContext.register(AppConfig.class);applicationContext.refresh();}
}

再次运行,就报错了:

image.png

需要注意的是,我们不能这么写:

AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext(AppConfig.class);applicationContext.setAllowCircularReferences(false);

如果你这么写,程序执行完第一行代码,整个Spring容器已经初始化完成了,你再设置不允许循环依赖,也于事无补了。

可以循环依赖的神奇之处在哪

有很多小伙伴可能并不觉得可以循环依赖有多么神奇,那是因为不知道矛盾点在哪,接下来就来说说这个问题:
当beanA,beanB循环依赖:

  1. 创建beanA,发现依赖beanB;
  2. 创建beanB,发现依赖beanA;
  3. 创建beanA,发现依赖beanB;
  4. 创建beanB,发现依赖beanA。
    ...
    好了,死循环了。

循环依赖的矛盾点就在于要创建beanA,它需要beanB,而创建beanB,又需要beanA,然后两个bean都创建不出来。

如何简单的解决循环依赖

如果你曾经看过Spring解决循环依赖的博客,应该知道它其中有好几个Map,一个Map放的是最完整的对象,称为singletonObjects,一个Map放的是提前暴露出来的对象,称为earlySingletonObjects。

在这里,先要解释下这两个东西:

  • singletonObjects:单例池,其中存放的是经历了Spring完整生命周期的bean,这里面的bean的依赖都已经填充完毕了。
  • earlySingletonObjects:提前暴露出来的对象的map,其中存放的是刚刚创建出来的对象,没有经历Spring完整生命周期的bean,这里面的bean的依赖还未填充完毕。

我们可以这么做:

  1. 当我们创建完beanA,就把自己放到earlySingletonObjects,发现自己需要beanB,然后就去屁颠屁颠创建beanB;
  2. 当我们创建完beanB,就把自己放到earlySingletonObjects,发现自己需要beanA,然后就去屁颠屁颠创建beanA;
  3. 创建beanA前,先去earlySingletonObjects看一下,发现自己已经被创建出来了,把自己返回出去;
  4. beanB拿到了beanA,beanB创建完毕,把自己放入singletonObjects;
  5. beanA可以去singletonObjects拿到beanB了,beanA也创建完毕,把自己放到singletonObjects。
    整个过程结束。

下面让我们来实现这个功能:
首先,自定义一个注解,字段上打上这个注解的,说明需要被Autowired:

@Retention(RetentionPolicy.RUNTIME)
public @interface CodeBearAutowired {
}

再创建两个循环依赖的类:

public class OrderService {@CodeBearAutowiredpublic UserService userService;
}
public class UserService {@CodeBearAutowiredpublic OrderService orderService;
}

然后就是核心,创建对象,填充属性,并解决Spring循环依赖的问题:

public class Cycle {// 单例池,里面放的是完整的bean,已完成填充属性private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();// 存放的是提前暴露出来的bean,没有经历过spring完整的生命周期,没有填充属性private final Map<String, Object> earlySingletonObjects = new HashMap<>();// 在Spring中,这个map存放的是beanNam和beanDefinition的映射关系static Map<String, Class<?>> map = new HashMap<>();static {map.put("orderService", OrderService.class);map.put("userService", UserService.class);}// 如果先调用init方法,就是预加载,如果直接调用getBean就是懒加载,两者的循环依赖问题都解决了public void init() {for (Map.Entry<String, Class<?>> stringClassEntry : map.entrySet()) {createBean(stringClassEntry.getKey());}}public Object getBean(String beanName) {// 尝试从singletonObjects中取,Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject != null) {return singletonObject;}// 尝试从earlySingletonObjects取singletonObject = this.earlySingletonObjects.get(beanName);if (singletonObject != null) {return singletonObject;}return createBean(beanName);}private Object createBean(String beanName) {Object singletonObject;try {// 创建对象singletonObject = map.get(beanName).getConstructor().newInstance();// 把没有完成填充属性的半成品 bean 放入earlySingletonObjectsearlySingletonObjects.put(beanName, singletonObject);// 填充属性populateBean(singletonObject);// bean创建成功,放入singletonObjectsthis.singletonObjects.put(beanName, singletonObject);return singletonObject;} catch (Exception ignore) {}return null;}private void populateBean(Object object) {Field[] fields = object.getClass().getDeclaredFields();for (Field field : fields) {if (field.getAnnotation(CodeBearAutowired.class) != null) {Object value = getBean(field.getName());try {field.setAccessible(true);field.set(object, value);} catch (IllegalAccessException ignored) {}}}}
}

预加载调用:

public class Main {public static void main(String[] args) {Cycle cycle = new Cycle();cycle.init();UserService userService = (UserService) cycle.getBean("userService");OrderService orderService = (OrderService) cycle.getBean("orderService");System.out.println(userService.orderService);System.out.println(orderService.userService);}
}

运行结果:

com.codebear.cycleeasy.OrderService@61baa894
com.codebear.cycleeasy.UserService@b065c63

懒加载调用:

public class Main {public static void main(String[] args) {Cycle cycle = new Cycle();UserService userService = (UserService) cycle.getBean("userService");OrderService orderService = (OrderService) cycle.getBean("orderService");System.out.println(userService.orderService);System.out.println(orderService.userService);}
}

运行结果:

com.codebear.cycleeasy.OrderService@61baa894
com.codebear.cycleeasy.UserService@b065c63

为什么无法解决原型、构造方法注入的循环依赖

在上面,我们自己手写了解决循环依赖的代码,可以看到,核心是利用一个map,来解决这个问题的,这个map就相当于缓存。

为什么可以这么做,因为我们的bean是单例的,而且是字段注入(setter注入)的,单例意味着只需要创建一次对象,后面就可以从缓存中取出来,字段注入,意味着我们无需调用构造方法进行注入。

  • 如果是原型bean,那么就意味着每次都要去创建对象,无法利用缓存;
  • 如果是构造方法注入,那么就意味着需要调用构造方法注入,也无法利用缓存。

需要aop怎么办?

我们上面的方案看起来很美好,但是还有一个问题,如果我们的bean创建出来,还要做一点加工,怎么办?也许,你没有理解这句话的意思,再说的明白点,如果beanA和【beanB的代理对象】循环依赖,或者【beanA的代理对象】和beanB循环依赖,再或者【beanA的代理对象】和【beanB的代理对象】循环依赖,怎么办?

这里说的创建代理对象仅仅是“加工”的其中一种可能。

遇到这种情况,我们总不能把创建完的对象直接扔到缓存把?我们这么做的话,如果【beanA的代理对象】和【beanB的代理对象】循环依赖,我们最终获取的beanA中的beanB还是beanB,并非是beanB的代理对象。

聪明的你,一定在想,这还不简单吗:
我们创建完对象后,判断这个对象是否需要代理,如果需要代理,创建代理对象,然后把代理对象放到earlySingletonObjects不就OJ8K了?
就像这样:

private Object createBean(String beanName) {Object singletonObject;try {// 创建对象singletonObject = map.get(beanName).getConstructor().newInstance();// 创建bean的代理对象/*** if( 需要代理){*     singletonObject=创建代理对象;** }*/// 把没有完成填充属性的半成品 bean 放入earlySingletonObjectsearlySingletonObjects.put(beanName, singletonObject);// 填充属性populateBean(singletonObject);// bean创建成功,放入singletonObjectsthis.singletonObjects.put(beanName, singletonObject);return singletonObject;} catch (Exception ignore) {}return null;}

这确实可以,但是,这违反了Spring的初衷,Spring的初衷是希望在bean生命周期的最后几步才去aop,如果像上面说的这么做,就意味着一旦创建完对象,Spring就会去aop了,这就违反了Spring的初衷,所以Spring并没有这么做。

但是如果真的出现了aop bean循环依赖,就没办法了,只能先去aop,但是如果没有出现循环依赖,Spring并不希望在这里就进行aop,所以Spring引入了Map<String, ObjectFactory<?>>,ObjectFactory是一个函数式接口,可以理解为工厂方法,当创建完对象后,把【获得这个对象的工厂方法】放入这个map,等真的发生循环依赖,就去执行这个【获得这个对象的工厂方法】,获取加工完成的对象。

下面直接放出代码:

public class Cycle {// 单例池,里面放的是完整的bean,已完成填充属性private final Map<String, Object> singletonObjects = new ConcurrentHashMap<>();// 存放的是 加工bean的工厂方法private final Map<String, ObjectFactory<?>> singletonFactories = new HashMap<>();// 存放的是提前暴露出来的bean,没有经历过spring完整的生命周期,没有填充属性private final Map<String, Object> earlySingletonObjects = new HashMap<>();private final Set<String> singletonsCurrentlyInCreation = new HashSet<>();static Map<String, Class<?>> map = new HashMap<>();static {map.put("orderService", OrderService.class);map.put("userService", UserService.class);}public void init() {for (Map.Entry<String, Class<?>> stringClassEntry : map.entrySet()) {createBean(stringClassEntry.getKey());}}private Object createBean(String beanName) {Object instance = null;try {instance = map.get(beanName).getConstructor().newInstance();} catch (Exception ex) {}Object finalInstance = instance;this.singletonFactories.put(beanName, () -> {// 创建代理对象return finalInstance;});populateBean(instance);this.singletonObjects.put(beanName, instance);return instance;}public Object getBean(String beanName) {// 尝试从singletonObjects中取,Object singletonObject = this.singletonObjects.get(beanName);if (singletonObject != null) {return singletonObject;}// 尝试从earlySingletonObjects取singletonObject = this.earlySingletonObjects.get(beanName);if (singletonObject != null) {return singletonObject;}// 尝试从singletonFactories取出工厂方法ObjectFactory<?> objectFactory = this.singletonFactories.get(beanName);if (objectFactory != null) {singletonObject = objectFactory.getObject();this.earlySingletonObjects.put(beanName, singletonObject);return singletonObject;}return createBean(beanName);}private void populateBean(Object object) {Field[] fields = object.getClass().getDeclaredFields();for (Field field : fields) {if (field.getAnnotation(CodeBearAutowired.class) != null) {Object value = getBean(field.getName());try {field.setAccessible(true);field.set(object, value);} catch (IllegalAccessException ignored) {}}}}
}

调用方法:

public static void main(String[] args) {Cycle cycle = new Cycle();cycle.init();System.out.println(((UserService) cycle.getBean("userService")).orderService);System.out.println(((OrderService) cycle.getBean("orderService")).userService);}

运行结果:

com.codebear.cycles.OrderService@49e4cb85
com.codebear.cycles.UserService@2133c8f8

二级缓存能不能解决循环依赖,三级循环到底有什么用?

我的观点可能和网上的主流观点有很大的出入,至于我的观点是对是错,请各位自行判断。

二级缓存可以解决循环依赖,哪怕aop bean循环依赖,上面我们已经提到了,我们可以创建完对象,直接创建代理对象,把代理对象放入二级缓存,这样我们从二级缓存获得的一定是aop bean,并非是bean本身。

三级缓存有什么用?网上的主流观点是为了解决循环依赖,还有就是为了效率,为了解决循环依赖,我们上面已经讨论过了,我的观点是二级缓存已经可以解决循环依赖了,下面就让我们想想,和效率是否有关系?

我的观点是没有关系,理由如下:
我们把【获得对象的工厂方法】放入了map

  • 如果没有循环依赖,这个map根本没有用到,和效率没有关系;
  • 如果是普通bean循环依赖,三级缓存直接返回了bean,和效率还是没有关系;
  • 如果是aop bean循环依赖,如果没有三级缓存,直接创建代理对象,放入二级缓存,如果有三级缓存,还是需要创建代理对象,只是两者的时机不同,和效率还是没有关系。

有了这篇博客的基础,当你再看其他关于Spring循环依赖的博客,应该会轻松的多,因为我们毕竟自己解决了循环依赖,Spring的循环依赖只是在我们之上做了进一步的封装与改进。


http://www.taodudu.cc/news/show-2895974.html

相关文章:

  • Spring当中循环依赖很少有人讲,今天让我们来看看吧
  • spring循环依赖让你更好的理解spring!!
  • 探针一号的SQL注入学习笔记
  • 微信获取nickname mysql乱码_微信nickname乱码(emoji)及mysql编码格式设置(utf8mb4)解决的过程...
  • 第四阶段微服务
  • 队列的基本操作
  • 高级版的 jvisualvm :Spring Boot Admin 监控 Spring Boot 微服务项目
  • java的“看门狗”锁续期可以用php redis这样实现【php锁续期、分布式锁、无锁请求队列超卖】解决【商家超卖(商品库存控制)、用户超买(秒杀订单控制)】问题。非demo 线上一直在用
  • 九度oj 题目1080:进制转换
  • 【杭电oj】2089 - 不要62(打表)
  • Jarvis OJ BASIC部分题目writeup
  • GitHub标星8k!java正则数字
  • Springboot WEBOJ项目—— Nginx部署
  • 杭电OJ2015
  • JarvisOJ Basic部分题目wp
  • 基于Halcon学习的一维码识别【十三】ean13.hdev
  • 如何把EAN13码外观设置和code128码一样
  • EAN13条形码了解,有c测试代码
  • php ean13,php生成EAN
  • 安卓生成EAN13
  • EAN 13码编码
  • php ean-13,用php生成EAN_13标准的条形码_php
  • php ean13,php生成EAN_13标准条形码实例_php实例
  • php ean13,php生成EAN_13标准条形码实例_PHP
  • BARCODE条形码插件及ean13
  • EAN13条形码绘制(Delphi版)
  • php ean13,php生成EAN_13标准条形码实例_PHP教程
  • python ean13条形码的验证_(转)基于SQL的EAN13、ENA8条形码校验位生成
  • 生成EAN13条码(C语言实现)
  • ean13解码 matlab,ean13: EAN13 Code Generator

帮助你更好的理解Spring循环依赖相关推荐

  1. Java开发常见面试题详解(LockSupport,AQS,Spring循环依赖,Redis)

    总览 问题 详解 String.intern()的作用 link LeetCode的Two Sum题 link 什么是可重入锁? link 谈谈LockSupport link 谈谈AQS link ...

  2. Java开发常见面试题详解(LockSupport,AQS,Spring循环依赖,Redis)_3

    Java开发常见面试题详解(LockSupport,AQS,Spring循环依赖,Redis)_3 总览 问题 详解 String.intern()的作用 link LeetCode的Two Sum题 ...

  3. spring循环依赖让你更好的理解spring!!

    什么是循环依赖 一言以蔽之:两者相互依赖. 在开发中,可能经常出现这种情况,只是我们平时并没有注意到原来我们写的两个类.甚至多个类相互依赖了,为什么注意不到呢?当然是因为没有报错,而且一点问题都木有, ...

  4. 这个Spring循环依赖的坑,90%以上的人都不知道

    点击上方"方志朋",选择"设为星标" 回复"666"获取新整理的面试资料 作者:Mythsman 原文:https://blog.myths ...

  5. 【spring容器启动】之bean的实例化和初始化(文末附:spring循环依赖原理)

    本次我们通过源码介绍ApplicationContext容器初始化流程,主要介绍容器内bean的实例化和初始化过程.ApplicationContext是Spring推出的先进Ioc容器,它继承了旧版 ...

  6. 终于有人把 Spring 循环依赖讲清楚了!

    网上关于Spring循环依赖的博客太多了,有很多都分析的很深入,写的很用心,甚至还画了时序图.流程图帮助读者理解,我看了后,感觉自己是懂了,但是闭上眼睛,总觉得还没有完全理解,总觉得还有一两个坎过不去 ...

  7. 这个 Spring 循环依赖的坑,90% 以上的人都不知道

    点击上方"后端技术精选",选择"置顶公众号" 技术文章第一时间送达! 作者:Mythsman blog.mythsman.com/post/5d838c7c2d ...

  8. Spring 循环依赖(5.2.x)

    目录 1.回顾getBean过程 2.回顾创建拥有简单依赖关系Bean的过程(类A里依赖类B) 3.Spring是如何解决循环依赖的 4.Spring Bean的代理对象是在什么时候创建的 5.AOP ...

  9. 烂了大街的 Spring 循环依赖问题,你以为自己就真会了吗

    前言 循环依赖问题,算是一道烂大街的面试题了,解毒之前,我们先来回顾两个知识点: 初学 Spring 的时候,我们就知道 IOC,控制反转么,它将原本在程序中手动创建对象的控制权,交由 Spring ...

最新文章

  1. 软件包有未满足的依赖关系
  2. java date 转换calendar_Java中日期格式(String、Date、Calendar)的相互转换
  3. u8 附件上传后存放路径_织梦DedeCms附件按月份保存的修改方法
  4. 【2017年第4期】大数据平台的基础能力和性能测试
  5. C#语法中String与string的区别
  6. 易语言数据类型与c 对照,易语言利用自定义数据类型和数组. 制作键对值操作类/内存配置...
  7. Docker监控方案之cAdvisor
  8. [译]GLUT教程 - 整合代码3
  9. x-admin发异步把数据提交给php,role-add.html
  10. python 删除大表数据
  11. 人证核验一体如何验证考生身份对比
  12. python 折线图变成直线图_python如何画折线图
  13. CP速配app v2.0.8.2
  14. bzoj2121 字符串游戏
  15. 电缆压降损耗计算方法
  16. 码云与与GitHUB的对比及使用码云使用细节
  17. Android MediaPlayer源码分析
  18. Eviews 8.0&9.0界面新功能介绍
  19. cocos2dx 3.0 Google使用obb扩展包
  20. 供应链管理系统(Java+SSH+MyEclipse+MySQL)

热门文章

  1. python onenet_onenet简介
  2. 现货期货价格关系(现货期货价格关系分析)
  3. 名帖291 董其昌 行书《乐志论》
  4. 这篇文章告诉你三个能给视频去水印的软件
  5. MAC结合LINUX进行Jmeter进行java代码压力测试
  6. java中金额转换精度问题
  7. Linux内核网络学习
  8. 【Rust日报】 2019-02-23
  9. 少儿编程教培管理系统,scratch课程
  10. 微软预览word_如何在Microsoft Word中更改语言