帮助你更好的理解Spring循环依赖

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

什么是循环依赖

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

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

如下所示:

@Configuration
@ComponentScan
public class AppConfig {
}
@Service
public class AuthorService {
    @Autowired
    BookService bookService;
}
@Service
public class BookService {
    @Autowired
    AuthorService 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 {
    @Autowired
    AuthorService authorService;
}
@Service
@Scope(BeanDefinition.SCOPE_PROTOTYPE)
public class AuthorService {
    @Autowired
    BookService bookService;
}

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

帮助你更好的理解Spring循环依赖

如果是构造参数注入的循环依赖,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;
    }
}

还是讨厌的红色字体:

帮助你更好的理解Spring循环依赖

循环依赖可以关闭吗

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

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

再次运行,就报错了:

帮助你更好的理解Spring循环依赖

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

        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 {
    @CodeBearAutowired
    public UserService userService;
}
public class UserService {
    @CodeBearAutowired
    public OrderService orderService;
}

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

public class Cycle {
    // 单例池,里面放的是完整的bean,已完成填充属性
    private final Map<string, object> singletonObjects = new ConcurrentHashMap<>();

    // &#x5B58;&#x653E;&#x7684;&#x662F;&#x63D0;&#x524D;&#x66B4;&#x9732;&#x51FA;&#x6765;&#x7684;bean&#xFF0C;&#x6CA1;&#x6709;&#x7ECF;&#x5386;&#x8FC7;spring&#x5B8C;&#x6574;&#x7684;&#x751F;&#x547D;&#x5468;&#x671F;&#xFF0C;&#x6CA1;&#x6709;&#x586B;&#x5145;&#x5C5E;&#x6027;
    private final Map<string, object> earlySingletonObjects = new HashMap<>();

    // &#x5728;Spring&#x4E2D;&#xFF0C;&#x8FD9;&#x4E2A;map&#x5B58;&#x653E;&#x7684;&#x662F;beanNam&#x548C;beanDefinition&#x7684;&#x6620;&#x5C04;&#x5173;&#x7CFB;
    static Map<string, class<?>> map = new HashMap<>();
    static {
        map.put("orderService", OrderService.class);
        map.put("userService", UserService.class);
    }
    // &#x5982;&#x679C;&#x5148;&#x8C03;&#x7528;init&#x65B9;&#x6CD5;&#xFF0C;&#x5C31;&#x662F;&#x9884;&#x52A0;&#x8F7D;&#xFF0C;&#x5982;&#x679C;&#x76F4;&#x63A5;&#x8C03;&#x7528;getBean&#x5C31;&#x662F;&#x61D2;&#x52A0;&#x8F7D;&#xFF0C;&#x4E24;&#x8005;&#x7684;&#x5FAA;&#x73AF;&#x4F9D;&#x8D56;&#x95EE;&#x9898;&#x90FD;&#x89E3;&#x51B3;&#x4E86;
    public void init() {
        for (Map.Entry<string, class<?>> stringClassEntry : map.entrySet()) {
            createBean(stringClassEntry.getKey());
        }
    }

    public Object getBean(String beanName) {
        // &#x5C1D;&#x8BD5;&#x4ECE;singletonObjects&#x4E2D;&#x53D6;&#xFF0C;
        Object singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject != null) {
            return singletonObject;
        }

        // &#x5C1D;&#x8BD5;&#x4ECE;earlySingletonObjects&#x53D6;
        singletonObject = this.earlySingletonObjects.get(beanName);
        if (singletonObject != null) {
            return singletonObject;
        }

        return createBean(beanName);
    }

    private Object createBean(String beanName) {
        Object singletonObject;

        try {
            // &#x521B;&#x5EFA;&#x5BF9;&#x8C61;
            singletonObject = map.get(beanName).getConstructor().newInstance();

            // &#x628A;&#x6CA1;&#x6709;&#x5B8C;&#x6210;&#x586B;&#x5145;&#x5C5E;&#x6027;&#x7684;&#x534A;&#x6210;&#x54C1; bean &#x653E;&#x5165;earlySingletonObjects
            earlySingletonObjects.put(beanName, singletonObject);

            // &#x586B;&#x5145;&#x5C5E;&#x6027;
            populateBean(singletonObject);

            // bean&#x521B;&#x5EFA;&#x6210;&#x529F;&#xFF0C;&#x653E;&#x5165;singletonObjects
            this.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) {
                }
            }
        }
    }
}
</string,></string,></string,></string,>

预加载调用:

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 {
            // &#x521B;&#x5EFA;&#x5BF9;&#x8C61;
            singletonObject = map.get(beanName).getConstructor().newInstance();

            // &#x521B;&#x5EFA;bean&#x7684;&#x4EE3;&#x7406;&#x5BF9;&#x8C61;
            /**
             * if( &#x9700;&#x8981;&#x4EE3;&#x7406;){
             *     singletonObject=&#x521B;&#x5EFA;&#x4EE3;&#x7406;&#x5BF9;&#x8C61;;
             *
             * }
             */

            // &#x628A;&#x6CA1;&#x6709;&#x5B8C;&#x6210;&#x586B;&#x5145;&#x5C5E;&#x6027;&#x7684;&#x534A;&#x6210;&#x54C1; bean &#x653E;&#x5165;earlySingletonObjects
            earlySingletonObjects.put(beanName, singletonObject);

            // &#x586B;&#x5145;&#x5C5E;&#x6027;
            populateBean(singletonObject);

            // bean&#x521B;&#x5EFA;&#x6210;&#x529F;&#xFF0C;&#x653E;&#x5165;singletonObjects
            this.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

下面直接放出代码:

public class Cycle {
    // &#x5355;&#x4F8B;&#x6C60;&#xFF0C;&#x91CC;&#x9762;&#x653E;&#x7684;&#x662F;&#x5B8C;&#x6574;&#x7684;bean&#xFF0C;&#x5DF2;&#x5B8C;&#x6210;&#x586B;&#x5145;&#x5C5E;&#x6027;
    private final Map<string, object> singletonObjects = new ConcurrentHashMap<>();

    // &#x5B58;&#x653E;&#x7684;&#x662F; &#x52A0;&#x5DE5;bean&#x7684;&#x5DE5;&#x5382;&#x65B9;&#x6CD5;
    private final Map<string, objectfactory<?>> singletonFactories = new HashMap<>();

    // &#x5B58;&#x653E;&#x7684;&#x662F;&#x63D0;&#x524D;&#x66B4;&#x9732;&#x51FA;&#x6765;&#x7684;bean&#xFF0C;&#x6CA1;&#x6709;&#x7ECF;&#x5386;&#x8FC7;spring&#x5B8C;&#x6574;&#x7684;&#x751F;&#x547D;&#x5468;&#x671F;&#xFF0C;&#x6CA1;&#x6709;&#x586B;&#x5145;&#x5C5E;&#x6027;
    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, () -> {
            // &#x521B;&#x5EFA;&#x4EE3;&#x7406;&#x5BF9;&#x8C61;
            return finalInstance;
        });

        populateBean(instance);

        this.singletonObjects.put(beanName, instance);
        return instance;
    }

    public Object getBean(String beanName) {
        // &#x5C1D;&#x8BD5;&#x4ECE;singletonObjects&#x4E2D;&#x53D6;&#xFF0C;
        Object singletonObject = this.singletonObjects.get(beanName);
        if (singletonObject != null) {
            return singletonObject;
        }

        // &#x5C1D;&#x8BD5;&#x4ECE;earlySingletonObjects&#x53D6;
        singletonObject = this.earlySingletonObjects.get(beanName);
        if (singletonObject != null) {
            return singletonObject;
        }

        // &#x5C1D;&#x8BD5;&#x4ECE;singletonFactories&#x53D6;&#x51FA;&#x5DE5;&#x5382;&#x65B9;&#x6CD5;
        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) {
                }
            }
        }
    }
}
</string,></string,></string></string,></string,></string,>

调用方法:

 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的循环依赖只是在我们之上做了进一步的封装与改进。

Original: https://www.cnblogs.com/CodeBear/p/13327899.html
Author: CodeBear
Title: 帮助你更好的理解Spring循环依赖

原创文章受到原创版权保护。转载请注明出处:https://www.johngo689.com/572154/

转载文章受原作者版权保护。转载请注明原作者出处!

(0)

大家都在看

  • InnoDB学习(七)之索引结构

    索引是对数据库表中一列或多列的值进行排序的一种结构,使用索引可快速访问数据库表中的特定信息。可以将数据库索引和书的目录进行类比,通过书的目录我们可以快速查找到章节位置,如果没有目录…

    Java 2023年6月8日
    073
  • HIT软构博客5–LAB2记录与总结

    本次实验我学习了ADT的设计、规约、测试,并使用OOP技术实现 ADT。 ​ 首先按照给定的需求,从中根据名词找到对应需要设计的ADT,然后确定ADT内所需要的方法,设计方法的sp…

    Java 2023年6月5日
    085
  • Spring系列9:基于注解的Spring容器配置

    前面几篇中我们说过,Spring容器支持3种方式进行bean定义信息的配置,现在具体说明下: XML:bean的定义和依赖都在xml文件中配置,比较繁杂。 Annotation-b…

    Java 2023年6月5日
    065
  • golang学习链接

    GitHub入门: https://github.com/rubyhan1314/Golang-100-Days Golang中国: https://www.qfgolang.co…

    Java 2023年6月6日
    074
  • SpringBoot的自动配置实现和介绍

    自动配置实现逻辑 –> 约定大于配置 在spring4.0时提出了 Condition相关注解, Condition相关注解可以让用户提供一个判断条件,从而返回t…

    Java 2023年5月30日
    068
  • 设计模式之组合模式

    本文通过老王和小王探讨书房、书架、各类书的管理问题,引出结构型设计模式家族中的一个重要成员——组合模式,本文会给予两种组合模式的典型代码实现,为了加深理解会在第三部分应用中介绍组合…

    Java 2023年6月8日
    079
  • 匿名内部类的使用

    原始做法: /** * 创建时间: 2022/4/30 * * @author 尼古拉斯.Cai灯 */ public class BasicClass { public stat…

    Java 2023年6月9日
    073
  • 深入MySQL(三):MySQL的索引的应用

    在MySQL的优化中,索引的作用绝对算是一个大头,很多时候索引使用得当可以使得一个查询的效率提高几个数量级,同时它还具有自动排序等功能。所以如果是深入MySQL,那么索引绝对是其中…

    Java 2023年6月7日
    061
  • 【AS3代码】擦窗户效果(也就是流行的妄撮游戏)

    源码下载:《 妄撮 (Mosatsu)》撕掉她的衣服在线 游戏 package{import flash.display.Bitmap;import flash.display.L…

    Java 2023年5月29日
    0116
  • 记一次Oracle session数过多引起进程数超过processes限制最终导致客户端无法连接的问题

    RAC突然告警,客户端尝试连接时会报:ORA-12520: TNS:listener could not find available handler for requested …

    Java 2023年5月30日
    045
  • eclipse中项目jdk1.8刷新下就变成1.5的解决办法

    https://blog.csdn.net/qq_32814555/article/details/82803017 在项目的pom.xml 文件中加上 Original: htt…

    Java 2023年5月30日
    077
  • 水淼·文件批量处理器

    水淼软件出品的一款批量改名、批量替换内容等综合批处理工具。 批量修改文件内容、属性、时间、快捷方式信息、音乐标签,批量创建文件和文件夹、智能归档,批量分割和合并文件,批量下载网络图…

    Java 2023年5月30日
    064
  • rocketmq有序消息

    RocketMQ提供的顺序消费消息实现是使用的FIFO 先进先出算法 Producer消息发送 public class Producer { public static void…

    Java 2023年6月16日
    058
  • 程序的输出结果是什么?

    #include using namespace std; int main() { int b=3; int arr[]={6,7,8,9,10}; int *ptr=arr; …

    Java 2023年6月7日
    071
  • SpringBoot:Springboot项目中调用第三方jar包中的类时报错–注入类的两种方式

    场景:将一些公共的方法封装成了一个jar包,在其他项目中进行引用的时候报错 报错原因:bean没有注入,引进来的jar包没有被spring管理, 因为类没有被@Service,@R…

    Java 2023年5月30日
    067
  • java几种常见的排序

    排序,就是将一串数组(一个列表)中的元素(整数,数字,字符串等)按某种顺序(增大,减小,字典顺序等)重新排列。 下面介绍几种排序 1.冒泡排序 定义:冒泡排序就是从第一个元素开始,…

    Java 2023年6月8日
    073
亲爱的 Coder【最近整理,可免费获取】👉 最新必读书单  | 👏 面试题下载  | 🌎 免费的AI知识星球