(上)解析ソースSpring5 5- ConfigurationClassPostProcessor

背面に接続され、我々は、の話refresh()方法のinvokeBeanFactoryPostProcessors(beanFactory)主な実装のメソッドBeanFactoryPostProcessorとそのサブインタフェースのBeanDefinitionRegistryPostProcessorメソッド。

作成AnnotationConfigApplicationContextオブジェクトの春のことは非常に重要な追加されますBeanFactoryPostProcessorインターフェースクラスを:ConfigurationClassPostProcessorただコンテナに追加追加することをここで注意してくださいbeanDefinitionMap、それはまた、豆の真の例を作成されていません。

簡単なレビューはConfigurationClassPostProcessorいつでもコンテナに追加する:AnnotationConfigApplicationContext引数なしのコンストラクタの作成はAnnotatedBeanDefinitionReader、オブジェクトに渡されるBeanDefinitionRegistry分析ノートを登録したクラスに関連するプロセッサの構成をBeanDefinitionConfigurationClassPostProcessorそれがここにコンテナに追加します。


ConfigurationClassPostProcessor

いくつかを見てくださいConfigurationClassPostProcessor継承システム:

ConfigurationClassPostProcessor

ConfigurationClassPostProcessor実装BeanDefinitionRegistryPostProcessorインターフェイスを、それがコンテナの登録にSpringコンテナで開始持つBeanDefinition機能を。

我々はことを知っているConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry方法がある中でrefresh();のプロセスinvokeBeanFactoryPostProcessors(beanFactory);が実行され、ここでは、この方法を見てみましょう。

ConfigurationClassPostProcessor#postProcessBeanDefinitionRegistry

@Override
public void postProcessBeanDefinitionRegistry(BeanDefinitionRegistry registry) {
    int registryId = System.identityHashCode(registry);
    if (this.registriesPostProcessed.contains(registryId)) {
        throw new IllegalStateException(
                "postProcessBeanDefinitionRegistry already called on this post-processor against " + registry);
    }
    if (this.factoriesPostProcessed.contains(registryId)) {
        throw new IllegalStateException(
                "postProcessBeanFactory already called on this post-processor against " + registry);
    }
    this.registriesPostProcessed.add(registryId);

    processConfigBeanDefinitions(registry);
}

メインロジックprocessConfigBeanDefinitions(registry);、オープンソースのポイント:

public void processConfigBeanDefinitions(BeanDefinitionRegistry registry) {
    List<BeanDefinitionHolder> configCandidates = new ArrayList<>();
    //获取所有的BeanDefinitionName
    String[] candidateNames = registry.getBeanDefinitionNames();

    for (String beanName : candidateNames) {
        BeanDefinition beanDef = registry.getBeanDefinition(beanName);

        // https://docs.spring.io/spring/docs/5.1.8.RELEASE/spring-framework-reference/core.html#beans-java-basic-concepts
        // Full @Configuration vs “lite” @Bean mode
        if (ConfigurationClassUtils.isFullConfigurationClass(beanDef) ||
                ConfigurationClassUtils.isLiteConfigurationClass(beanDef)) {
            if (logger.isDebugEnabled()) {
                logger.debug("Bean definition has already been processed as a configuration class: " + beanDef);
            }
        }

        // 校验是否为配置类
        // 配置类分为两种 Full @Configuration vs “lite” @Bean mode
        // 校验之后在 BeanDefinition 中添加标志属性
        // 如果满足条件则加入到configCandidates
        else if (ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef, this.metadataReaderFactory)) {
            // 如果是配置类,就放到 configCandidates 变量中
            configCandidates.add(new BeanDefinitionHolder(beanDef, beanName));
        }
    }

    // Return immediately if no @Configuration classes were found
    if (configCandidates.isEmpty()) {
        return;
    }

    // Sort by previously determined @Order value, if applicable
    configCandidates.sort((bd1, bd2) -> {
        int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());
        int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());
        return Integer.compare(i1, i2);
    });

    // Detect any custom bean name generation strategy supplied through the enclosing application context
    SingletonBeanRegistry sbr = null;
    // 传入的 registry 是 DefaultListableBeanFactory
    if (registry instanceof SingletonBeanRegistry) {
        sbr = (SingletonBeanRegistry) registry;
        if (!this.localBeanNameGeneratorSet) {
            //获取自定义BeanNameGenerator,一般情况下为空
            BeanNameGenerator generator = (BeanNameGenerator) sbr.getSingleton(CONFIGURATION_BEAN_NAME_GENERATOR);
            if (generator != null) {
                this.componentScanBeanNameGenerator = generator;
                this.importBeanNameGenerator = generator;
            }
        }
    }

    if (this.environment == null) {
        this.environment = new StandardEnvironment();
    }

    // Parse each @Configuration class
    // new ConfigurationClassParser,用来解析 @Configuration 类
    ConfigurationClassParser parser = new ConfigurationClassParser(
            this.metadataReaderFactory, this.problemReporter, this.environment,
            this.resourceLoader, this.componentScanBeanNameGenerator, registry);

    // 将 configCandidates 转成 set  candidates , 去重
    Set<BeanDefinitionHolder> candidates = new LinkedHashSet<>(configCandidates);
    Set<ConfigurationClass> alreadyParsed = new HashSet<>(configCandidates.size());
    do {
        // 解析配置类
        parser.parse(candidates);
        parser.validate();

        Set<ConfigurationClass> configClasses = new LinkedHashSet<>(parser.getConfigurationClasses());
        configClasses.removeAll(alreadyParsed);

        // Read the model and create bean definitions based on its content
        if (this.reader == null) {
            this.reader = new ConfigurationClassBeanDefinitionReader(
                    registry, this.sourceExtractor, this.resourceLoader, this.environment,
                    this.importBeanNameGenerator, parser.getImportRegistry());
        }
        // Import类,@Bean,@ImportResource 转化为 BeanDefinition
        this.reader.loadBeanDefinitions(configClasses);
        alreadyParsed.addAll(configClasses);

        candidates.clear();
        // 再获取一下容器中BeanDefinition的数据,如果发现数量增加了,说明有新的BeanDefinition被注册了
        if (registry.getBeanDefinitionCount() > candidateNames.length) {
            String[] newCandidateNames = registry.getBeanDefinitionNames();
            Set<String> oldCandidateNames = new HashSet<>(Arrays.asList(candidateNames));
            Set<String> alreadyParsedClasses = new HashSet<>();
            for (ConfigurationClass configurationClass : alreadyParsed) {
                alreadyParsedClasses.add(configurationClass.getMetadata().getClassName());
            }
            for (String candidateName : newCandidateNames) {
                if (!oldCandidateNames.contains(candidateName)) {
                    BeanDefinition bd = registry.getBeanDefinition(candidateName);
                    if (ConfigurationClassUtils.checkConfigurationClassCandidate(bd, this.metadataReaderFactory) &&
                            !alreadyParsedClasses.contains(bd.getBeanClassName())) {
                        candidates.add(new BeanDefinitionHolder(bd, candidateName));
                    }
                }
            }
            candidateNames = newCandidateNames;
        }
    }
    while (!candidates.isEmpty());

    // Register the ImportRegistry as a bean in order to support ImportAware @Configuration classes
    if (sbr != null && !sbr.containsSingleton(IMPORT_REGISTRY_BEAN_NAME)) {
        sbr.registerSingleton(IMPORT_REGISTRY_BEAN_NAME, parser.getImportRegistry());
    }

    if (this.metadataReaderFactory instanceof CachingMetadataReaderFactory) {
        // Clear cache in externally provided MetadataReaderFactory; this is a no-op
        // for a shared cache since it'll be cleared by the ApplicationContext.
        ((CachingMetadataReaderFactory) this.metadataReaderFactory).clearCache();
    }
}

それは、構成クラス判定された場合は、すべてのBeanDefinitionNames、ループ配列を取得します。

candidateNames

トップ5の春の登録はプロセッサに内蔵されて、最後の一つはに渡されAnnotationConfigApplicationContextた構成クラスAppConfig.class

2 ConfigurationClassは1であり、春にあり、FullConfigurationClass他のですLiteConfigurationClass両者の違いについて、あなたはの@Beanモードliteの全@Configuration上の記事や記事を見ることができる前に。

ConfigurationClassUtils#checkConfigurationClassCandidate内部カテゴリに属する構成決定方法、及びBeanDefinitionタグ決意結果。次のように特定決意ロジックがあります。

public static boolean isFullConfigurationCandidate(AnnotationMetadata metadata) {
    return metadata.isAnnotated(Configuration.class.getName());
}
private static final Set<String> candidateIndicators = new HashSet<>(8);

static {
    candidateIndicators.add(Component.class.getName());
    candidateIndicators.add(ComponentScan.class.getName());
    candidateIndicators.add(Import.class.getName());
    candidateIndicators.add(ImportResource.class.getName());
}

public static boolean isLiteConfigurationCandidate(AnnotationMetadata metadata) {
    // Do not consider an interface or an annotation...
    if (metadata.isInterface()) {
        return false;
    }

    // Any of the typical annotations found?
    for (String indicator : candidateIndicators) {
        if (metadata.isAnnotated(indicator)) {
            return true;
        }
    }

    // Finally, let's look for @Bean methods...
    try {
        return metadata.hasAnnotatedMethods(Bean.class.getName());
    } catch (Throwable ex) {
        if (logger.isDebugEnabled()) {
            logger.debug("Failed to introspect @Bean methods on class [" + metadata.getClassName() + "]: " + ex);
        }
        return false;
    }
}

分析configCandidates格納された変数

空でない場合は、設定クラスが空であるかどうか、そして、それらを並べ替えます。

configCandidates

作成しConfigurationClassParser解析するためのオブジェクトを@Configuration、スキャンパッケージを完了するために、クラスをBeanDefinition登録します。実施を通じてparser.parse(candidates);メソッドを完了します。

実行parser.parse(candidates)方法の前に:

実行parser.parse前(候補)メソッド

実装parser.parse(candidates)方法:

parser.parse(候補)メソッドを実行した後

コンフィギュレーション・クラスの構文解析が完了した後、直ちにの実施後this.reader.loadBeanDefinitions(configClasses);方法。このメソッドは、主に対処するために使用されImport类@Beanおよび@ImportResource注釈。これらの2つの方法についての具体的な詳細は、我々は次の時間を再訪します。

そして最後まで添加ImportAwareに必要な豆のインタフェースをサポート。

// Register the ImportRegistry as a bean in order to support ImportAware @Configuration classes
if (sbr != null && !sbr.containsSingleton(IMPORT_REGISTRY_BEAN_NAME)) {
    sbr.registerSingleton(IMPORT_REGISTRY_BEAN_NAME, parser.getImportRegistry());
}

ImportAwareインターフェイスの使用、我々はその次の時間を繰り返すことができます。


継続するには......

ソースの研究ノートます。https://github.com/shenjianeng/spring-code-study

私達の注意にパブリック番号を歓迎し、私たちは一緒に勉強し成長します。
コーダ黒

おすすめ

転載: www.cnblogs.com/coderxiaohei/p/11685144.html