spring ioc源码解析(bean定义加载)

最近学习了一段时间spring 源码,写一些学习笔记,大家互相交流一下。

spring 容器的启动过程,主要做了两件事情,beanDefinition的加载 和 bean的创建

首先我们来看下beanFactory 接口的继承图
beanFactory继承图.png

在接下来的源码分析中,我们分析两个类 AnnotationConfigApplicationContext, DefaultListableBeanFactory

AnnotationConfigApplicationContext这个类是基于注解的容器类,它实现了BeanFactory和BeanDefinitionRegistry两个接口,拥有bean对象管理和beanDefinition注册的功能,同时在这个类里边还拥有一个DefaultListableBeanFactory 对象。

DefaultListableBeanFactory 这个类也实现了BeanFactory和BeanDefinitionRegistry两个接口,拥有bean对象管理和beanDefinition注册的功能,AnnotationConfigApplicationContext是委托DefaultListableBeanFactory来实现bean对象管理和beanDefinition的注册。

beanDefinition的加载

beanDefition个是一个接口,是用来存储Bean定义的一些信息的,比如ClassName,Scope,init-methon,等等。它的实现有RootBeanDefination,AnnotatedGenericBeanDefinition等。

首先我们通过一个小的demo,来启动一个spring容器。

  public static void main(String[] args) {
        AnnotationConfigApplicationContext cxt = new AnnotationConfigApplicationContext(MainConfig.class);
        Car car = (Car) cxt.getBean("car");
        System.out.println(car.getName());
    }

  @Configuration
  @Import(value = {TimeProxy.class,LogProxy.class})
  @ComponentScan(value = "com.thread.proyx")
  public class MainConfig {
    @Bean
    public Car car(){
        return new Car();
    }
    @Bean
    public Tran tran(){
        return new Tran();
    }
  }

接下来我们对源码进行分析,创建AnnotationConfigApplicationContext对象

 public AnnotationConfigApplicationContext(Class... componentClasses) {
        //加载spring内部类的beanDefinition到容器中
        this();
        //加载配置类的beanDefinition
        this.register(componentClasses);
        //加载其它类beanDefinition,创建bean实例化...
        this.refresh();
    }

this构造方法,给我们创建了AnnotatedBeanDefinitionReader注解bean定义读取,ClassPathBeanDefinitionScanner 类路径bean定义扫描器,还有它的父类构造方法,创建了 DefaultListableBeanFactory 对象,

public AnnotationConfigApplicationContext() {

    this.reader = new AnnotatedBeanDefinitionReader(this);

    this.scanner = new ClassPathBeanDefinitionScanner(this);
}
//父类构造器
public GenericApplicationContext() {

    this.customClassLoader = false;

    this.refreshed = new AtomicBoolean();

//创建beanFactory

    this.beanFactory = new DefaultListableBeanFactory();

}

AnnotatedBeanDefinitionReader对象的创建,是用来加载spring内部类的beanDefinition,我们从源码可以看出

 public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment) {
        //省略一些代码...
        //注册注解形式的配置处理器
        AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);
    }

 public static Set<BeanDefinitionHolder> registerAnnotationConfigProcessors(BeanDefinitionRegistry registry, @Nullable Object source) {
        //判断registry是否为空
        DefaultListableBeanFactory beanFactory = unwrapDefaultListableBeanFactory(registry);
       //省略配置beanFactory的一些代码

        Set<BeanDefinitionHolder> beanDefs = new LinkedHashSet(8);
        RootBeanDefinition def;
        //ConfigurationClassPostProcessor类型的beanDefinition注册到容器中,这个类型很重要,接下来,我们自己注入的类,通过扫描器扫描的类,解析方法就在这个类里边。
        if (!registry.containsBeanDefinition("org.springframework.context.annotation.internalConfigurationAnnotationProcessor")) {
            def = new RootBeanDefinition(ConfigurationClassPostProcessor.class);
            def.setSource(source);
            beanDefs.add(registerPostProcessor(registry, def, "org.springframework.context.annotation.internalConfigurationAnnotationProcessor"));
        }
        //解析autowired自动装配的beanDefinition注册到容器中(这是个后置处理器)
        if (!registry.containsBeanDefinition("org.springframework.context.annotation.internalAutowiredAnnotationProcessor")) {
            def = new RootBeanDefinition(AutowiredAnnotationBeanPostProcessor.class);
            def.setSource(source);
            beanDefs.add(registerPostProcessor(registry, def, "org.springframework.context.annotation.internalAutowiredAnnotationProcessor"));
        }
        //省略了一些代码(spring内部需要注册到容器中的beanDefinition)
        return beanDefs;
    }

到这里,我们的AnnotatedBeanDefinitionReader对象就创建完成了,AnnotatedBeanDefinitionReader对象的创建过程,就是将spring内部的一些处理器的beanDefinition加载到容器中。

ClassPathBeanDefinitionScanner(类路径bean定义扫描器)对象的创建,就是创建默认的扫描器,代码比较简单。this方法已经完成,主要就是加载spring内部类的beanDefinition到容器中。

registe方法,传入的参数,就是我们创建的配置类,这个方法是把我们配置类的beanDefinition加载到容器中

    public void register(Class... componentClasses) {
        Assert.notEmpty(componentClasses, "At least one component class must be specified");
        //this方法创建的reader对象,调用register方法,传入配置类
        this.reader.register(componentClasses);
    }

    public void register(Class... componentClasses) {
        Class[] var2 = componentClasses;
        int var3 = componentClasses.length;
        for(int var4 = 0; var4 < var3; ++var4) {
            Class<?> componentClass = var2[var4];
        //调用注册配置类的beanDefinition
            this.registerBean(componentClass);
        }
    }

   public void registerBean(Class<?> beanClass) {
        this.doRegisterBean(beanClass, (String)null, (Class[])null, (Supplier)null, (BeanDefinitionCustomizer[])null);
    }

    
    private <T> void doRegisterBean(Class<T> beanClass, @Nullable String name, @Nullable Class<? extends Annotation>[] qualifiers, @Nullable Supplier<T> supplier, @Nullable BeanDefinitionCustomizer[] customizers) {
        AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(beanClass);
         //省略一些代码
            //创建beanDefinition的包装类
            BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);
            definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
            //继续调用注册bean定义方法
            BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);
        }
    }

    public static void registerBeanDefinition(BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry) throws BeanDefinitionStoreException {
        String beanName = definitionHolder.getBeanName();
        //BeanDefinitionRegistry 是接口,调用方法实现类的方法,前面说到DefaultListableBeanFactory 实现了该接口,直接调用 DefaultListableBeanFactory  的registerBeanDefinition 方法
        registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());
        String[] aliases = definitionHolder.getAliases();
        //配置别名
        if (aliases != null) {
            String[] var4 = aliases;
            int var5 = aliases.length;
            for(int var6 = 0; var6 < var5; ++var6) {
                String alias = var4[var6];
                registry.registerAlias(beanName, alias);
            }
        }
    }

//在 DefaultListableBeanFactory的 registerBeanDefinition方法中有端代码

this.beanDefinitionMap.put(beanName, beanDefinition);

说明了把beanDefinition保存在 map中,beanName为key,beanDefinition为value;这里是把配置类的bean定义注册到容器中;到这里 registe方法就完成了,当前只是把配置类的bean定义注册完成。我们自己注入的bean,扫描的bean定义还没有加载。

AnnotationConfigApplicationContext的最后一个方法 refresh,这里边是注册@bean,@ComponentScan, @Import 等注入的beanDefinition,创建bean实例等操作。

 public void refresh() throws BeansException, IllegalStateException {
        synchronized(this.startupShutdownMonitor) {
            this.prepareRefresh();
            ConfigurableListableBeanFactory beanFactory = this.obtainFreshBeanFactory();
            this.prepareBeanFactory(beanFactory);
            try {
                this.postProcessBeanFactory(beanFactory);
                //这个方法很重要,其它beanDefinition注册,都在这里边实现,本篇接下来主要介绍这个方法的实现。
                this.invokeBeanFactoryPostProcessors(beanFactory);
                //注册bean后置处理器,aop用到
                this.registerBeanPostProcessors(beanFactory);
                this.initMessageSource();
                //初始化事件多播器,事件监听使用
                this.initApplicationEventMulticaster();
                this.onRefresh();
                //事件监听器
                this.registerListeners();
                //这个方法很重要,完成bean实例化的创建,包括解决循环依赖,这篇里边不做介绍,
                this.finishBeanFactoryInitialization(beanFactory);
                this.finishRefresh();
            } catch (BeansException var9) {
                if (this.logger.isWarnEnabled()) {
                    this.logger.warn("Exception encountered during context initialization - cancelling refresh attempt: " + var9);
                }
                this.destroyBeans();
                this.cancelRefresh(var9);
                throw var9;
            } finally {
                this.resetCommonCaches();
            }
        }
    }

接下来看下我们来看invokeBeanFactoryPostProcessors(beanFactory)方法里边的源码

    protected void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory) {
        //传入参数beanFactory和 beanFactory后置处理器
        PostProcessorRegistrationDelegate.invokeBeanFactoryPostProcessors(beanFactory, this.getBeanFactoryPostProcessors());
        //省略一些代码
    }

    public static void invokeBeanFactoryPostProcessors(ConfigurableListableBeanFactory beanFactory, List<BeanFactoryPostProcessor> beanFactoryPostProcessors) {
            //其它省略的代码,一些看不懂,
            //这个方法是最主要的处理方法,继续调用,还没有进去到处理的地方
            invokeBeanDefinitionRegistryPostProcessors(currentRegistryProcessors, registry);
            currentRegistryProcessors.clear();
           
    private static void invokeBeanDefinitionRegistryPostProcessors(Collection<? extends BeanDefinitionRegistryPostProcessor> postProcessors, BeanDefinitionRegistry registry) {
        Iterator var2 = postProcessors.iterator();
        while(var2.hasNext()) {
            BeanDefinitionRegistryPostProcessor postProcessor = (BeanDefinitionRegistryPostProcessor)var2.next();
            //这里就是调用ConfigurationClassPostProcessor类里边的方法,ConfigurationClassPostProcessor就是刚开始注册的那个很重要的内部bean定义
            postProcessor.postProcessBeanDefinitionRegistry(registry);
        }
    }

//这里就是ConfigurationClassPostProcessor的方法
  public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {
          //省略一些代码
            this.registriesPostProcessed.add(registryId);
            //这里才是处理配置bean定义的主要方法,接着往下看
            this.processConfigBeanDefinitions(registry);
    }

    public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
             //省略一些配置的代码
            //创建配置类的解析器
            ConfigurationClassParser parser = new ConfigurationClassParser(this.metadataReaderFactory, this.problemReporter, this.environment, this.resourceLoader, this.componentScanBeanNameGenerator, registry);
            Set<BeanDefinitionHolder> candidates = new LinkedHashSet(configCandidates);
            HashSet alreadyParsed = new HashSet(configCandidates.size());
            do {
                //解析配置类里边@ComponentScan 注解扫描的bean,加载beanDefinition到容器中,并找到@bean,@Import 等注解引入的bean
                parser.parse(candidates);
                //把上面找到但没有加载的bean定义,加载到容器中
                this.reader.loadBeanDefinitions(configClasses);
              }while(!candidates.isEmpty())
               
    }

接下来我们看下parse方法里边的源码,

   public void parse(Set<BeanDefinitionHolder> configCandidates) {
        Iterator var2 = configCandidates.iterator();
        while(var2.hasNext()) {
            BeanDefinitionHolder holder = (BeanDefinitionHolder)var2.next();
            BeanDefinition bd = holder.getBeanDefinition();
            try {
                if (bd instanceof AnnotatedBeanDefinition) {
                //注解的bean定义,从这个方法进入
                    this.parse(((AnnotatedBeanDefinition)bd).getMetadata(), holder.getBeanName());
                } else if (bd instanceof AbstractBeanDefinition && ((AbstractBeanDefinition)bd).hasBeanClass()) {
                    this.parse(((AbstractBeanDefinition)bd).getBeanClass(), holder.getBeanName());
                } else {
                    this.parse(bd.getBeanClassName(), holder.getBeanName());
                }
            } catch (BeanDefinitionStoreException var6) {
                throw var6;
            } catch (Throwable var7) {
                throw new BeanDefinitionStoreException("Failed to parse configuration class [" + bd.getBeanClassName() + "]", var7);
            }
        }
        this.deferredImportSelectorHandler.process();
    }

   protected final void parse(AnnotationMetadata metadata, String beanName) throws IOException {
        this.processConfigurationClass(new ConfigurationClass(metadata, beanName), DEFAULT_EXCLUSION_FILTER);
    }

    protected void processConfigurationClass(ConfigurationClass configClass, Predicate<String> filter) throws IOException {
            do {
              //这里是spring做事的处理方法
                sourceClass = this.doProcessConfigurationClass(configClass, sourceClass, filter);
            } while(sourceClass != null);
            this.configurationClasses.put(configClass, configClass);
    }
    @Nullable
    protected final ConfigurationClassParser.SourceClass doProcessConfigurationClass(ConfigurationClass configClass, ConfigurationClassParser.SourceClass sourceClass, Predicate<String> filter) throws IOException {
      
        // 处理 @ComponentScan 注解类,重点,先获取加了@ComponentScan注解的配置类
        Set<AnnotationAttributes> componentScans = AnnotationConfigUtils.attributesForRepeatable(sourceClass.getMetadata(), ComponentScans.class, ComponentScan.class);
        if (!componentScans.isEmpty() && !this.conditionEvaluator.shouldSkip(sourceClass.getMetadata(), ConfigurationPhase.REGISTER_BEAN)) {
            Iterator var14 = componentScans.iterator();
            while(var14.hasNext()) {
                AnnotationAttributes componentScan = (AnnotationAttributes)var14.next();
                 // 开始扫描指定package下的所有Bean,那些加了@Component,@Service,@Repository等注解的类,都会加进来。
                Set<BeanDefinitionHolder> scannedBeanDefinitions = this.componentScanParser.parse(componentScan, sourceClass.getMetadata().getClassName());
                Iterator var8 = scannedBeanDefinitions.iterator();
                while(var8.hasNext()) {
                    BeanDefinitionHolder holder = (BeanDefinitionHolder)var8.next();
                    BeanDefinition bdCand = holder.getBeanDefinition().getOriginatingBeanDefinition();
                    if (bdCand == null) {
                        bdCand = holder.getBeanDefinition();
                    }
                    if (ConfigurationClassUtils.checkConfigurationClassCandidate(bdCand, this.metadataReaderFactory)) {
                        this.parse(bdCand.getBeanClassName(), holder.getBeanName());
                    }
                }
            }
        }
            //查询import导入的beanDefinition
        this.processImports(configClass, sourceClass, this.getImports(sourceClass), filter, true);
        importResource = AnnotationConfigUtils.attributesFor(sourceClass.getMetadata(), ImportResource.class);
        if (importResource != null) {
            String[] resources = importResource.getStringArray("locations");
            Class<? extends BeanDefinitionReader> readerClass = importResource.getClass("reader");
            String[] var20 = resources;
            int var22 = resources.length;

            for(int var23 = 0; var23 < var22; ++var23) {
                String resource = var20[var23];
                String resolvedResource = this.environment.resolveRequiredPlaceholders(resource);
                configClass.addImportedResource(resolvedResource, readerClass);
            }
        }
         //查询@bean导入的beanDefinition
        Set<MethodMetadata> beanMethods = this.retrieveBeanMethodMetadata(sourceClass);
        Iterator var18 = beanMethods.iterator();

        while(var18.hasNext()) {
            MethodMetadata methodMetadata = (MethodMetadata)var18.next();
            configClass.addBeanMethod(new BeanMethod(methodMetadata, configClass));
        }

        this.processInterfaces(configClass, sourceClass);
        if (sourceClass.getMetadata().hasSuperClass()) {
            String superclass = sourceClass.getMetadata().getSuperClassName();
            if (superclass != null && !superclass.startsWith("java") && !this.knownSuperclasses.containsKey(superclass)) {
                this.knownSuperclasses.put(superclass, configClass);
                return sourceClass.getSuperClass();
            }
        }

        return null;
    }


    public Set<BeanDefinitionHolder> parse(AnnotationAttributes componentScan, final String declaringClass) {
        //创建扫描器
        ClassPathBeanDefinitionScanner scanner = new ClassPathBeanDefinitionScanner(this.registry, componentScan.getBoolean("useDefaultFilters"), this.environment, this.resourceLoader);
        //省略代码
        //开始扫描,将扫描的BeanDefinitionHolder返回
        return scanner.doScan(StringUtils.toStringArray(basePackages));
    }

protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
        Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();
        for (String basePackage : basePackages) {
            // 找出当前包下面的所有符合条件的类,这个方法我们就不分析了。
            Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
            for (BeanDefinition candidate : candidates) {
                // 处理一些其他情况,也不说了...
                if (candidate instanceof AbstractBeanDefinition) {
                    postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
                }
                if (candidate instanceof AnnotatedBeanDefinition) {
                    AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
                }
                if (checkCandidate(beanName, candidate)) {
                    BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
                    definitionHolder =
                            AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
                    beanDefinitions.add(definitionHolder);
                    // 最后注册BeanDefinition,这样就将包下的所有BeanDefintion都注册完成了
                    registerBeanDefinition(definitionHolder, this.registry);
                }
            }
        }
        return beanDefinitions;
    }

到这里为止,我们配置类的分析就结束了,将扫描的包下的BeanDefinition都注册完成了,接下来我们看下其它方式导入的beanDefinition的注册方法
this.reader.loadBeanDefinitions(configClasses);

private void loadBeanDefinitionsForConfigurationClass(
            ConfigurationClass configClass, TrackedConditionEvaluator trackedConditionEvaluator) {
        // 处理Imported的配置类
        if (configClass.isImported()) {
            registerBeanDefinitionForImportedConfigurationClass(configClass);
        }
        for (BeanMethod beanMethod : configClass.getBeanMethods()) {
            // 注册所有的@Bean方法上的类,重点看一下这个方法
            loadBeanDefinitionsForBeanMethod(beanMethod);
        }

        // 处理其它注解,这里就不分析了。
        loadBeanDefinitionsFromImportedResources(configClass.getImportedResources());
                 loadBeanDefinitionsFromRegistrars(configClass.getImportBeanDefinitionRegistrars());
    }

private void loadBeanDefinitionsForBeanMethod(BeanMethod beanMethod) {
        ConfigurationClass configClass = beanMethod.getConfigurationClass();        
        // 一些别名处理的代码都省略了...

        //  创建BeanDefinition
        ConfigurationClassBeanDefinition beanDef = new ConfigurationClassBeanDefinition(configClass, metadata);

        // 处理initMethod和destroyMethod
        String initMethodName = bean.getString("initMethod");
        if (StringUtils.hasText(initMethodName)) {
            beanDef.setInitMethodName(initMethodName);
        }

        String destroyMethodName = bean.getString("destroyMethod");
        beanDef.setDestroyMethodName(destroyMethodName);        

        // Replace the original bean definition with the target one, if necessary
        BeanDefinition beanDefToRegister = beanDef;
        // 最后注册这个BeanDefinition
        this.registry.registerBeanDefinition(beanName, beanDefToRegister);
    }

到这里,所有的beanDefinition全部注册完成

总结
spring 注册beanDefinition,首先是注册spring内部的bean,接着是注册我们编写的配置类,最后我们通过对配置类的扫描,加载扫描的所有beanDefinition。
我也是第一次写这么多东西,如果有哪些地方有问题,还希望跟各位多多交流。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,186评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,858评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,620评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,888评论 1 285
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,009评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,149评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,204评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,956评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,385评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,698评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,863评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,544评论 4 335
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,185评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,899评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,141评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,684评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,750评论 2 351