Spring循环依赖-spring源码详解(四)

简介: Spring循环依赖-spring源码详解(四)
 public static void loadBeanDefinitions() {
        RootBeanDefinition rootBeanDefinitionA = new RootBeanDefinition(InstanceA.class);
        RootBeanDefinition rootBeanDefinitionB = new RootBeanDefinition(InstanceB.class);
        beanDefinitionMap.put("instanceA", rootBeanDefinitionA);
        beanDefinitionMap.put("instanceB", rootBeanDefinitionB);
    }

@Component
public class InstanceA {
    @Autowired
    private InstanceB instanceB;

    public InstanceB getInstanceB() {
        return instanceB;
    }

    public void setInstanceB(InstanceB instanceB) {
        this.instanceB = instanceB;
    }

    public InstanceA(InstanceB instanceB) {
        this.instanceB = instanceB;
    }

    public InstanceA() {
        System.out.println("实例化A");
    }
}

首先把bean放入beanDefinition,之后,循环bean定义,通过bean的key来获取beanDefinition,

通过无参构造函数反射来获取class,再赋值属性。

        // 加载到beanDefinition
        loadBeanDefinitions();

        for (String key : beanDefinitionMap.keySet()) {
            // 先创建A
            getBean(key);
        }
        InstanceA instanceA = (InstanceA)getBean("instanceA");
        System.out.println(instanceA);
    }

    private static Map<String, Object> singletonObjects = new ConcurrentHashMap<>();

    public static Object getBean(String beanName) throws Exception {
        Object singleton = getSingleton(beanName);
        if(singleton != null){
            return singleton;
        }

        // 实例化
        RootBeanDefinition rootBeanDefinition = (RootBeanDefinition) beanDefinitionMap.get(beanName);
        Class<?> beanClass = rootBeanDefinition.getBeanClass();
        // 通过class 无参构造函数 实例化
        Object instanceBean = beanClass.newInstance();

        singletonObjects.put(beanName, instanceBean);
        // 属性赋值
        Field[] declaredFields = beanClass.getDeclaredFields();
        for (Field declaredField : declaredFields) {
            Autowired annotation = declaredField.getAnnotation(Autowired.class);
            // 说明bean有autowired
            if (annotation != null) {
                // 打开访问权限
                declaredField.setAccessible(true);
                // byName  byType   拿到了 instanceB名字
                String name = declaredField.getName();
                // 拿到B的bean
                Object fileObject = getBean(name);
                declaredField.set(instanceBean, fileObject);
            }
        }
        // 初始化
        // 放入缓存
//        singletonObjects.put(beanName, instanceBean);
        return instanceBean;
    }

    private static Object getSingleton(String beanName) {
        Object singleton = singletonObjects.get(beanName);
        if (singleton != null) {
            return singleton;
        }
        return null;
    }

image.png

这样写完之后,就不会有死循环,但是没有用到二级缓存,二级缓存可以用来放半成品的bean,解决多线程的情况下,多个线程同时创建bean,防止其他线程拿到不完整的bean,于是 加上二级缓存来写。

   private static Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>();

   public static Object getBean(String beanName) throws Exception {
       Object singleton = getSingleton(beanName);
       if (singleton != null) {
           return singleton;
       }

       // 实例化
       RootBeanDefinition rootBeanDefinition = (RootBeanDefinition) beanDefinitionMap.get(beanName);
       Class<?> beanClass = rootBeanDefinition.getBeanClass();
       // 通过class 无参构造函数 实例化
       Object instanceBean = beanClass.newInstance();

       // 二级缓存
       earlySingletonObjects.put(beanName, instanceBean);
       // 属性赋值
       Field[] declaredFields = beanClass.getDeclaredFields();
       for (Field declaredField : declaredFields) {
           Autowired annotation = declaredField.getAnnotation(Autowired.class);
           // 说明bean有autowired
           if (annotation != null) {
               // 打开访问权限
               declaredField.setAccessible(true);
               // byName  byType   拿到了 instanceB名字
               String name = declaredField.getName();
               // 拿到B的bean
               Object fileObject = getBean(name);
               declaredField.set(instanceBean, fileObject);
           }
       }
       // 初始化
       // 放入缓存
       singletonObjects.put(beanName, instanceBean);
       return instanceBean;
   }

   private static Object getSingleton(String beanName) {
       Object singleton = singletonObjects.get(beanName);
       if (singleton != null) {
           return singleton;
       }
       Object earlySingleton = earlySingletonObjects.get(beanName);
       if (earlySingleton != null) {
           return earlySingleton;
       }
       return null;
   }

用解耦的方式,beanPostProcessor来创建动态代理。

如果在最后调用动态代理,这时候循环依赖的bean就不是动态代理的bean,所以要在之前就创建动态代理。所以这里有两个地方调用动态代理,实例化之后和初始化之后调用。

只在循环依赖的情况下在实例化之后创建动态代理,所以需要判断当前是不是循环依赖。
前面一级缓存中没拿到,而二级缓存中有,才是循环依赖。

而spring源码里有一行代码,用来判断是否是循环依赖,加了一个正在创建对象的标识。

   private static Map<String, Object> earlySingletonObjects = new ConcurrentHashMap<>();
   // 存放函数接口
   private static Map<String, ObjectFactory> singletonFactories = new ConcurrentHashMap<>();

   private static Set<String> singletonCurrennlyInCreation = new HashSet<>();

   public static Object getBean(String beanName) throws Exception {
       Object singleton = getSingleton(beanName);
       if (singleton != null) {
           return singleton;
       }

       // 正在创建
       if (!singletonCurrennlyInCreation.contains(beanName)) {
           singletonCurrennlyInCreation.add(beanName);
       }

       // 实例化
       RootBeanDefinition rootBeanDefinition = (RootBeanDefinition) beanDefinitionMap.get(beanName);
       Class<?> beanClass = rootBeanDefinition.getBeanClass();
       // 通过class 无参构造函数 实例化
       Object instanceBean = beanClass.newInstance();

       // 创建动态代理
       // 只在循环依赖的情况下在实例化之后创建动态代理
       //Object o = new JDKProxyBeanPostProcessor().getEarlyBeanReference(instanceBean,beanName);
       /*singletonFactories.put(beanName, new ObjectFactory() {
           @Override
           public Object getObject() throws BeansException {
               return  new JDKProxyBeanPostProcessor().getEarlyBeanReference(instanceBean,beanName);
           }
       });*/
       singletonFactories.put(beanName, () -> new JDKProxyBeanPostProcessor().getEarlyBeanReference(instanceBean, beanName));

       // 二级缓存
       //earlySingletonObjects.put(beanName, instanceBean);
       // 属性赋值
       Field[] declaredFields = beanClass.getDeclaredFields();
       for (Field declaredField : declaredFields) {
           Autowired annotation = declaredField.getAnnotation(Autowired.class);
           // 说明bean有autowired
           if (annotation != null) {
               // 打开访问权限
               declaredField.setAccessible(true);
               // byName  byType   拿到了 instanceB名字
               String name = declaredField.getName();
               // 拿到B的bean
               Object fileObject = getBean(name);
               declaredField.set(instanceBean, fileObject);
           }
       }
       // 初始化


       /*if(earlySingletonObjects.containsKey(beanName)){
           instanceBean = earlySingletonObjects.get(beanName);
       }*/
       // 放入缓存
       singletonObjects.put(beanName, instanceBean);
       return instanceBean;
   }

   private static Object getSingleton(String beanName) {
     /*  Object singleton = singletonObjects.get(beanName);
       if (singleton != null) {
           return singleton;
       }
       Object earlySingleton = earlySingletonObjects.get(beanName);
       if (earlySingleton != null) {

           return earlySingleton;
       }
       return null;*/
       Object bean = singletonObjects.get(beanName);
       if (bean == null && singletonCurrennlyInCreation.contains(beanName)) {
           // 是循环依赖
           bean  = earlySingletonObjects.get(beanName);
           synchronized (singletonObjects){
               if(bean == null){
                   ObjectFactory objectFactory = singletonFactories.get(beanName);
                   if (objectFactory != null) {
                       // 钩子函数 去创建 aop代理
                       bean = objectFactory.getObject();
                       earlySingletonObjects.put(beanName, bean);
                       singletonFactories.remove(beanName);
                   }
                   return bean;
               }
           }
       }
       return bean;
   }

相关文章
|
21天前
|
缓存 Java 开发工具
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
三级缓存是Spring框架里,一个经典的技术点,它很好地解决了循环依赖的问题,也是很多面试中会被问到的问题,本文从源码入手,详细剖析Spring三级缓存的来龙去脉。
Spring是如何解决循环依赖的?从底层源码入手,详细解读Spring框架的三级缓存
|
21天前
|
缓存 安全 Java
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
从底层源码入手,通过代码示例,追踪AnnotationConfigApplicationContext加载配置类、启动Spring容器的整个流程,并对IOC、BeanDefinition、PostProcesser等相关概念进行解释
Spring框架中Bean是如何加载的?从底层源码入手,详细解读Bean的创建流程
|
21天前
|
XML 缓存 Java
手写Spring源码(简化版)
Spring包下的类、手写@ComponentScan注解、@Component注解、@Autowired注解、@Scope注解、手写BeanDefinition、BeanNameAware、InitializingBean、BeanPostProcessor 、手写AnnotationConfigApplicationContext
手写Spring源码(简化版)
|
5天前
|
缓存 Java Spring
手写Spring Ioc 循环依赖底层源码剖析
在Spring框架中,IoC(控制反转)是一个核心特性,它通过依赖注入(DI)实现了对象间的解耦。然而,在实际开发中,循环依赖是一个常见的问题。
15 4
|
11天前
|
XML 缓存 Java
spring源码剖析-spring-beans(内部核心组件,BeanDefinition的注册,BeanWapper创建)
spring源码剖析-spring-beans(内部核心组件,BeanDefinition的注册,BeanWapper创建)
39 10
|
11天前
|
XML 存储 Java
spring源码刨析-spring-beans(内部核心组件,beanDefinition加载过程)
spring源码刨析-spring-beans(内部核心组件,beanDefinition加载过程)
|
10天前
|
XML 存储 Java
Spring-源码深入分析(二)
Spring-源码深入分析(二)
|
10天前
|
XML 设计模式 Java
Spring-源码深入分析(一)
Spring-源码深入分析(一)
|
2月前
|
人工智能 前端开发 Java
【实操】Spring Cloud Alibaba AI,阿里AI这不得玩一下(含前后端源码)
本文介绍了如何使用 **Spring Cloud Alibaba AI** 构建基于 Spring Boot 和 uni-app 的聊天机器人应用。主要内容包括:Spring Cloud Alibaba AI 的概念与功能,使用前的准备工作(如 JDK 17+、Spring Boot 3.0+ 及通义 API-KEY),详细实操步骤(涵盖前后端开发工具、组件选择、功能分析及关键代码示例)。最终展示了如何成功实现具备基本聊天功能的 AI 应用,帮助读者快速搭建智能聊天系统并探索更多高级功能。
594 2
【实操】Spring Cloud Alibaba AI,阿里AI这不得玩一下(含前后端源码)
|
2月前
|
存储 缓存 Java
面试问Spring循环依赖?今天通过代码调试让你记住
该文章讨论了Spring框架中循环依赖的概念,并通过代码示例帮助读者理解这一概念。
面试问Spring循环依赖?今天通过代码调试让你记住
下一篇
无影云桌面