Spring IOC(1)—-容器刷新(refresh())之前

  • 2019 年 10 月 6 日
  • 笔记

首先本次分析是基于注解形式的,想来xml格式的原理都是类似的。

首先说一下什么是Bean定义(beandefinition):bean定义并不是实例化的bean,而是bean对象的一些信息,根据这些定义的信息最后来初始化bean。比方说就是一些类的信息,还有我们配置的该实例的信息(是否懒加载,单例还是多例,作用域等等信息)。

进入正题:

 

 

AnnotationConfigApplicationContext构造方法传入一个主配置类,那么就从该构造方法进去。

 

 

一、我们先来看在容器刷新(refresh())之前,都干了什么

1.this()首先会调用无参构造,进而调用父类的无参构造,初始化beanFactory(DefaultListableBeanFactory),再者初始化注解模式下的bean定义读取器,classPath类型的bean定义扫描器;

 

2.register(annotatedClasses),基于上面初始化的bean定义读取器去解析注册我们自己的配置类信息到容器中。

 

具体的流程图如下:

 

 

 

完成了这些准备工作。refresh()里面就会实例化具体的bean了。

二、两个个扩展点

1. BeanDefinitionRegistryPostProcessor

Bean定义处理器,在注册bean定义的时候调用,可以自己注册新的bean定义

先直接通过上下文对象获取一个student,会报错,没有对应的bean定义

实现自己的Bean定义处理器

MyBeanDefinitionRegistryPostProcessor
package com.nijunyang.spring;    import com.nijunyang.spring.model.Student;  import org.springframework.beans.BeansException;  import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;  import org.springframework.beans.factory.support.BeanDefinitionRegistry;  import org.springframework.beans.factory.support.BeanDefinitionRegistryPostProcessor;  import org.springframework.beans.factory.support.RootBeanDefinition;  import org.springframework.stereotype.Component;    /**   * @author: create by nijunyang   * @date:2019/10/6   */  @Component  public class MyBeanDefinitionRegistryPostProcessor implements BeanDefinitionRegistryPostProcessor {      //bean定义注册处理器,在注册bean定义的时候调用,可以自己注册新的bean定义,比如Student      @Override      public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) throws BeansException {          System.out.println("bean定义的数据量:"+registry.getBeanDefinitionCount());          RootBeanDefinition rootBeanDefinition = new RootBeanDefinition(Student.class);          //注册一个名为student的bean定义进去          registry.registerBeanDefinition("student", rootBeanDefinition);          System.out.println("bean定义的数据量:"+registry.getBeanDefinitionCount());      }        @Override      public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {        }  }  

 注意需要加上@Component,并且主配置指定扫描的包将自己定义的类加载进去

MainConfig
package com.nijunyang.spring;    import org.springframework.context.annotation.ComponentScan;  import org.springframework.context.annotation.Configuration;    /**   * @author: create by nijunyang   * @date:2019/10/6   */  @Configuration  @ComponentScan(basePackages = "com.nijunyang.spring")  public class MainConfig {  }  


 

 

 正常获取bean。

 

2.BeanFactoryPostProcessor

BeanFactory后置处理器,注册了bean定义之后调用,可以修改bean定义,比如实现一个自己的BeanFactory后置处理器,把某个bean设置成懒加载。

 断点运行1步骤中的代码,发现student不是懒加载的

 

 实现自己的BeanFactory后置处理器,将student设置成懒加载

MyBeanFactoryPostProcessor
package com.nijunyang.spring;    import org.springframework.beans.BeansException;  import org.springframework.beans.factory.config.BeanDefinition;  import org.springframework.beans.factory.config.BeanFactoryPostProcessor;  import org.springframework.beans.factory.config.ConfigurableListableBeanFactory;  import org.springframework.stereotype.Component;    /**   * @author: create by nijunyang   * @date:2019/10/6   */  @Component  public class MyBeanFactoryPostProcessor implements BeanFactoryPostProcessor {      //注册了bean定义之后调用,可以修改bean定义,比如把student设置成懒加载      @Override      public void postProcessBeanFactory(ConfigurableListableBeanFactory beanFactory) throws BeansException {          for(String name : beanFactory.getBeanDefinitionNames()) {              if("student".equals(name)) {                  BeanDefinition beanDefinition = beanFactory.getBeanDefinition(name);                  beanDefinition.setLazyInit(true);              }            }      }  }  

 同样断点运行,发现student不会马上被初始化了