IOC初始化源码

概念

DI:依赖注入
IOC:控制反转

容器初始化

以注解的方式启动

配置类

@Configuration
@ComponentScan(basePackages = "com.text")
public class MainConfig {
   public MainConfig() {
      System.out.println("MainConfig");
   }
}

启动类

public class MainStat {
   public static void main(String[] args) {
      AnnotationConfigApplicationContext context= new AnnotationConfigApplicationContext(MainConfig.class);
   }
}

容器的初始化就通过new ApplicationContext()完成,下面来看具体做了些什么事情

初始化过程

进入到ApplicationContext()的构造函数中

{
    this();
    
    register();
    
    refresh();
}

一共调用了三个方法

首先来看this()

this调用构造函数,默认会调用父类的构造函数,这里GenericApplicationContext作为父类,父类的构造函数实例化了DefaultListableBeanFactory,为

ApplicationContext spring上下文对象初始化beanFactory
public GenericApplicationContext() {
    this.beanFactory = new DefaultListableBeanFactory();
}

回到AnnotationConfigApplicationContext构造器

public AnnotationConfigApplicationContext() {
   /**
    * 初始化注解模式下的bean定义扫描器
    * 调用AnnotatedBeanDefinitionReader构造方法,传入的是this(AnnotationConfigApplicationContext)对象
    */
   //注册内置BeanPostProcessor以及注册相关的BeanDefinition
   this.reader = new AnnotatedBeanDefinitionReader(this);
   /**
    * 初始化我们的classPath类型的bean定义扫描器
    */
   this.scanner = new ClassPathBeanDefinitionScanner(this);
}

第7行注册6个内置的beanPostProcessor

registry.registerBeanDefinition(beanName, definition);

然后this()就这样走完了,这里再总结一下this已经完成的工作

  • 初始化上下文对象beanFactory
  • 初始化bean定义扫描器BeanDefinitionReader,并且完成6个内置beanPostProcessor的注册
  • 初始化classPath类型的bean定义扫描器,也就是在外边显示调用context.scan(),添加入容器的bean

然后是我们的register()

这里传入了一个参数,本例是注解的方式启动,最开始将注解配置类作为参数传递进来,这里就作为register的参数。进去注册我们的配置类

一路调用,用我们刚刚初始化的bean定义扫描器reader
reader.register() --> for: registerBean(class) --> doRegisterBean(class,null,null,null)

spring的命名有一个规律,do开头的才是真正做事情的

ok,一路调用发现最终doRegisterBean在做事情

<T> void doRegisterBean(Class<T> annotatedClass, @Nullable Supplier<T> instanceSupplier, @Nullable String name,
      @Nullable Class<? extends Annotation>[] qualifiers, BeanDefinitionCustomizer... definitionCustomizers) {

   AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass);
   if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) {
      return;
   }

   abd.setInstanceSupplier(instanceSupplier);
   ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
   abd.setScope(scopeMetadata.getScopeName());
   String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));

   AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);
   if (qualifiers != null) {
      for (Class<? extends Annotation> qualifier : qualifiers) {
         if (Primary.class == qualifier) {
            abd.setPrimary(true);
         }
         else if (Lazy.class == qualifier) {
            abd.setLazyInit(true);
         }
         else {
            abd.addQualifier(new AutowireCandidateQualifier(qualifier));
         }
      }
   }
   for (BeanDefinitionCustomizer customizer : definitionCustomizers) {
      customizer.customize(abd);
   }

   BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
   definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
   BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
}

我们对这个方法逐步解析,首先是将配置类传入实例化了一个GenericBeanDefinition对象

AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass);

然后判断是否要跳过这个类解析,注解@Conditional

if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) {
      return;
   }

这里通过配置类构建一个scopeMetadata对象,将解析配置类上的@Scope注解,并将其value设置到abd对象中

ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);
   abd.setScope(scopeMetadata.getScopeName());

这里拿到beanName

String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));

处理注解bean的通用注解 lazy primary DependsOn Role Description,将解析结果set到abd中

AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);

这里是@Qualifier的处理

if (qualifiers != null) {
   for (Class<? extends Annotation> qualifier : qualifiers) {
      if (Primary.class == qualifier) {
         abd.setPrimary(true);
      }
      else if (Lazy.class == qualifier) {
         abd.setLazyInit(true);
      }
      else {
         abd.addQualifier(new AutowireCandidateQualifier(qualifier));
      }
   }
}

将adb和beanName作为参数实例化一个 deanDefinitionHolder,hodler相当于是一个bean的持有者或者说是bean的一个封装对象。

BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);

这里将配置类注册到beanDefinitionMap中

BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);

这里总结一下register方法,总的来说实质上就是将我们的配置类注入到bean定义map中,这个过程中能发现的是,在将配置类注入之前,首先我们通过配置类构建了一个beanDefinition对象,然后通过解析配置类上的一些注解,诸如@Scope,@Conditional,@lazy等,不断的将这个beanDefinition完善。实质上这个beanDefinition就是这个配置类的描述类,需要将它相关的描述写入。就好比这里是在画一个人的肖像,spring就是画家,bean就是人体模特,而这个beanDefinition就是这幅肖像画

最后是我们的refresh方法

首先整体来看一下里面的方法

@Override
public void refresh() throws BeansException, IllegalStateException {
   synchronized (this.startupShutdownMonitor) {
      //1:准备刷新上下文环境
      prepareRefresh();

      //2:获取告诉子类初始化Bean工厂,拿到DefaultListableBeanFactory实例对象
      ConfigurableListableBeanFactory beanFactory = obtainFreshBeanFactory();

      //3:对bean工厂进行填充属性
      prepareBeanFactory(beanFactory);

      try {
         // 第四:留个子类去实现该接口
         postProcessBeanFactory(beanFactory);

         // 调用我们的bean工厂的后置处理器,处理BeanFactoryPostProcessors接口的实现,bean实例化之前
         invokeBeanFactoryPostProcessors(beanFactory);

         // 调用我们bean的后置处理器,处理BeanPostProcessors接口的实现,bean实例化之后,流程同上一个方法类似
         registerBeanPostProcessors(beanFactory);

         // 初始化国际化资源处理器.
         initMessageSource();

         // 创建事件多播器
         initApplicationEventMulticaster();

         // 这个方法同样也是留个子类实现的springboot也是从这个方法进行启动tomcat的.
         onRefresh();

         //把我们的事件监听器注册到多播器上
         registerListeners();

         //实例化我们剩余的单实例bean.
         finishBeanFactoryInitialization(beanFactory);

         // 最后容器刷新 发布刷新事件(Spring cloud也是从这里启动的)
         finishRefresh();
      }

      catch (BeansException ex) {
         if (logger.isWarnEnabled()) {
            logger.warn("Exception  encountered during context initialization - " +
                  "cancelling refresh attempt: " + ex);
         }

         // Destroy already created singletons to avoid dangling resources.
         destroyBeans();

         // Reset 'active' flag.
         cancelRefresh(ex);

         // Propagate exception to caller.
         throw ex;
      }

      finally {
         // Reset common introspection caches in Spring's core, since we
         // might not ever need metadata for singleton beans anymore...
         resetCommonCaches();
      }
   }
}

从第18行开始,该方法的描述是调用我们的bean工厂的后置处理器,处理BeanFactoryPostProcessors接口的实现,发生在bean实例化之前
invokeBeanFactoryPostProcessors(beanFactory);

这里由于该方法的篇幅过长我们就简单描述一下该方法的作用,详细的参考另一篇笔记

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值