概述

在invokeBeanFactoryPostProcessors中,spring回对spring中默认的BeanFactory的后置处理器进行调用,而这里相关的最主要的类就是ConfigurationClassPostProcessor,ConfigurationClassPostProcessor这个类继承自BeanDefinitionRegistryPostProcessor,而BeanDefinitionRegistryPostProcessor继承自BeanFactoryPostProcessor
在这里,对BeanFactoryPostProcessor的postProcessBeanFactory和BeanDefinitionRegistryPostProcessor的postProcessBeanDefinitionRegistry都会进行调用

processConfigBeanDefinitions

在 postProcessBeanDefinitionRegistry() 这个方法中,做的事情都在 processConfigBeanDefinitions() 这个方法中,所以直接来到processConfigBeanDefinitions()

List<BeanDefinitionHolder> configCandidates = new ArrayList<>();
		String[] candidateNames = registry.getBeanDefinitionNames();

		for (String beanName : candidateNames) {
			BeanDefinition beanDef = registry.getBeanDefinition(beanName);
			if (beanDef.getAttribute(ConfigurationClassUtils.CONFIGURATION_CLASS_ATTRIBUTE) != null) {
				if (logger.isDebugEnabled()) {
					logger.debug("Bean definition has already been processed as a configuration class: " + beanDef);
				}
			}
			else if (ConfigurationClassUtils.checkConfigurationClassCandidate(beanDef, this.metadataReaderFactory)) {
				configCandidates.add(new BeanDefinitionHolder(beanDef, beanName));
			}
		}

		// Return immediately if no @Configuration classes were found
		// 没有找到配置类
		if (configCandidates.isEmpty()) {
			return;
		}

configCandidates :存放配置 bean 的集合
candidateNames :获取容器中注册的BeanDefinitioin
获取到容器中注册的BeanDefinition后就会,进行循环,对于循环的m每一个BeanDefintion会判断其是不是一个进行配置的Bean

下面来看看,Spring是如何判断一个Bean是不是配置bean

public static boolean checkConfigurationClassCandidate(
			BeanDefinition beanDef, MetadataReaderFactory metadataReaderFactory) {

		String className = beanDef.getBeanClassName();
		if (className == null || beanDef.getFactoryMethodName() != null) {
			return false;
		}

		AnnotationMetadata metadata;
		if (beanDef instanceof AnnotatedBeanDefinition &&
				className.equals(((AnnotatedBeanDefinition) beanDef).getMetadata().getClassName())) {
			metadata = ((AnnotatedBeanDefinition) beanDef).getMetadata();
		}
		//省略部分代码
		
		Map<String, Object> config = metadata.getAnnotationAttributes(Configuration.class.getName());
		if (config != null && !Boolean.FALSE.equals(config.get("proxyBeanMethods"))) {
			beanDef.setAttribute(CONFIGURATION_CLASS_ATTRIBUTE, CONFIGURATION_CLASS_FULL);
		}

		else if (config != null || isConfigurationCandidate(metadata)) {
			beanDef.setAttribute(CONFIGURATION_CLASS_ATTRIBUTE, CONFIGURATION_CLASS_LITE);
		}
		else {
			return false;
		}

		Integer order = getOrder(metadata);
		if (order != null) {
			beanDef.setAttribute(ORDER_ATTRIBUTE, order);
		}

		return true;
	}

当遍历到我们自己写的配置类的时候,就会进入第一个if(),这里就拿到配置类的原数据信息,例如我们写的配置类叫做AppConfig,就会拿到AppConfig这个类的元数据信息
接着,通过元数据信息就可以判断当前这个类上面有没有加 @Configruation 这个注解,有的话就会设置CONFIGURATION_CLASS_ATTRIBUTE这个属性为CONFIGURATION_CLASS_FULL
没有的话,还有一个判断 isConfigurationCandidate(metadata)

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 isConfigurationCandidate(AnnotationMetadata metadata) {
		if (metadata.isInterface()) {return false;}
		for (String indicator : candidateIndicators) {
			if (metadata.isAnnotated(indicator)) {
				return true;
			}
		}
		try {
			return metadata.hasAnnotatedMethods(Bean.class.getName());
		}
		catch (Throwable ex) {
			...
			return false;
		}
	}

首先是排除接口,然后判断如果类上面存在@Component,@ComponentScan,@Import,@ImportResource
或者,如果这个类里面 有方法加了@Bean,那么spring也会将这个类认为是一个配置类
就会将 CONFIGURATION_CLASS_ATTRIBUTE这个属性为CONFIGURATION_CLASS_LITE

在检查是不是一个配置类的最后还会进行BeanDefinition 中 ORDER_ATTRIBUTE属性的设置,这个属性就是解析配置Bean的顺序
这个值就是从元数据中获取@Order注解的value

configCandidates.sort((bd1, bd2) -> {
			int i1 = ConfigurationClassUtils.getOrder(bd1.getBeanDefinition());
			int i2 = ConfigurationClassUtils.getOrder(bd2.getBeanDefinition());
			return Integer.compare(i1, i2);
		});

完成上述后在 processConfigBeanDefinitions 这个方法里面就会根据 ORDER_ATTRIBUTE 进行排序

public static final String CONFIGURATION_BEAN_NAME_GENERATOR =
				"org.springframework.context.annotation.internalConfigurationBeanNameGenerator";
				
		if (registry instanceof SingletonBeanRegistry) {
			sbr = (SingletonBeanRegistry) registry;
			if (!this.localBeanNameGeneratorSet) {
				// 程序员可以指定一个beanName生成器 applicationContext.setBeanNameGenerator()
				BeanNameGenerator generator = (BeanNameGenerator) sbr.getSingleton(
						AnnotationConfigUtils.CONFIGURATION_BEAN_NAME_GENERATOR);
				if (generator != null) {
					this.componentScanBeanNameGenerator = generator;
					//@Import的bean是无法指定beanName的  所以这里有一个Import的BeanName的生成器
					this.importBeanNameGenerator = generator;
				}
			}
		}

这一步的操作是获取beanName的生成器,我们注册到spring 的bean常常没有指定beanName,所以spring需要自己取给Bean生成一个BeanName
这里spring就会去找org.springframework.context.annotation.internalConfigurationBeanNameGenerator这个类,然后赋值给componentScanBeanNameGenerator 和 importBeanNameGenerator

接下来就是最关键的部分了

do {
			// 对配置类进行解析
			parser.parse(candidates);
			parser.validate();

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

			if (this.reader == null) {
				this.reader = new ConfigurationClassBeanDefinitionReader(
						registry, this.sourceExtractor, this.resourceLoader, this.environment,
						this.importBeanNameGenerator, parser.getImportRegistry());
			}

			this.reader.loadBeanDefinitions(configClasses);
			alreadyParsed.addAll(configClasses);

			candidates.clear();
			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());

这部分的代码用了do-while的循环,因为spring在解析配置类的时候,很可能这个配置类又导入了另外一个配置类,所以需要进行迭代的解析
首先是parser.parse(candidates),对之前check出来的配置类进行解析,然后从解析出来的类里面再获取到配置类,并把这些类封装为ConfigurationClass对象放入set集合。而ConfigurationClass这个对象可以看成是配置类的信息,存储着一些配置类的元数据
当然,这里获取的配置类会包含之前解析过的配置类,所以扫描之后还会进行一次过滤,过滤到解析过的配置类

来看看spring具体是如何进行解析的

public void parse(Set<BeanDefinitionHolder> configCandidates) {
		for (BeanDefinitionHolder holder : configCandidates) {
			BeanDefinition bd = holder.getBeanDefinition();
			try {
				if (bd instanceof AnnotatedBeanDefinition) {
					parse(((AnnotatedBeanDefinition) bd).getMetadata(), holder.getBeanName());
				}
				else if (bd instanceof AbstractBeanDefinition && ((AbstractBeanDefinition) bd).hasBeanClass()) {
					parse(((AbstractBeanDefinition) bd).getBeanClass(), holder.getBeanName());
				}
				else {
					parse(bd.getBeanClassName(), holder.getBeanName());
				}
			}
			catch (BeanDefinitionStoreException ex) {
				....
			}
		}
		this.deferredImportSelectorHandler.process();
	}

configCandidates:check出来的配置类
这里虽然有三层的if else判断,但是做的事情都是同一个事情,获取当前配置类的元信息,然后进行parse()

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

	protected void processConfigurationClass(ConfigurationClass configClass) throws IOException {
		//删除非关键代码
		SourceClass sourceClass = asSourceClass(configClass);
		do {
			sourceClass = doProcessConfigurationClass(configClass, sourceClass);
		}
		while (sourceClass != null);
		this.configurationClasses.put(configClass, configClass);
	}

解析的核心就是再这个do-while 循环里面,这才是真正的解析,这个do-while的目的其实是去遍历父类
首先会解析当前配置配,解析完成后,就回去判断,这个类有没有父类,如果有父类就会将父类添加到sourceClass
再交给外层的do-while进行解析
然后解析完后会将这个配置Bean添加到configurationClasses

doProcessConfigurationClass

if (configClass.getMetadata().isAnnotated(Component.class.getName())) {
			processMemberClasses(configClass, sourceClass);
		}
	
	private void processMemberClasses(ConfigurationClass configClass, SourceClass sourceClass) throws IOException {
		Collection<SourceClass> memberClasses = sourceClass.getMemberClasses();
		if (!memberClasses.isEmpty()) {
			List<SourceClass> candidates = new ArrayList<>(memberClasses.size());
			for (SourceClass memberClass : memberClasses) {
				if (ConfigurationClassUtils.isConfigurationCandidate(memberClass.getMetadata()) &&
						!memberClass.getMetadata().getClassName().equals(configClass.getMetadata().getClassName())) {
					candidates.add(memberClass);
				}
			}
			OrderComparator.sort(candidates);
			for (SourceClass candidate : candidates) {
				if (this.importStack.contains(configClass)) {
					this.problemReporter.error(new CircularImportProblem(configClass, this.importStack));
				}
				else {
					this.importStack.push(configClass);
					try {
						processConfigurationClass(candidate.asConfigClass(configClass));
					}
					finally {
						this.importStack.pop();
					}
				}
			}
		}
	}

首先是对于@Componet这个注解的解析,如果加了@Componet,就会进入processMemberClasses()这个方法
在processMemberClasses()里面会去判断当前解析的类是不是有内部类,获取内部类集合,然后遍历内部类集合
判断内部类上是否存在Component、ComponentScan、Import、ImportResource注解,如果存,就说明这个内部类是一个配置类
就会递归进行处理

for (AnnotationAttributes propertySource : AnnotationConfigUtils.attributesForRepeatable(
				sourceClass.getMetadata(), PropertySources.class,
				org.springframework.context.annotation.PropertySource.class)) {
			if (this.environment instanceof ConfigurableEnvironment) {
				processPropertySource(propertySource);
			}
			else {
				...
			}
		}

解析@PropertySource注解,如果有这个注解,就会取出注解的值,然后调用processPropertySource()进行处理

private void processPropertySource(AnnotationAttributes propertySource) throws IOException {
		String name = propertySource.getString("name");
		if (!StringUtils.hasLength(name)) {
			name = null;
		}
		String encoding = propertySource.getString("encoding");
		if (!StringUtils.hasLength(encoding)) {
			encoding = null;
		}
		String[] locations = propertySource.getStringArray("value");
		Assert.isTrue(locations.length > 0, "At least one @PropertySource(value) location is required");
		boolean ignoreResourceNotFound = propertySource.getBoolean("ignoreResourceNotFound");

		Class<? extends PropertySourceFactory> factoryClass = propertySource.getClass("factory");
		PropertySourceFactory factory = (factoryClass == PropertySourceFactory.class ?
				DEFAULT_PROPERTY_SOURCE_FACTORY : BeanUtils.instantiateClass(factoryClass));

		// @PropertySource注解中所指定的properties文件路径
		for (String location : locations) {
			try {
				// 解析占位符
				String resolvedLocation = this.environment.resolveRequiredPlaceholders(location);
				// 得到资源文件
				Resource resource = this.resourceLoader.getResource(resolvedLocation);
				// 把资源文件解析成PropertySource对象,并且添加到environment中去
				addPropertySource(factory.createPropertySource(name, new EncodedResource(resource, encoding)));
			}
			catch (IllegalArgumentException | FileNotFoundException | UnknownHostException ex) {
				// Placeholders not resolvable or resource not found when trying to open it
				if (ignoreResourceNotFound) {
					if (logger.isInfoEnabled()) {
						logger.info("Properties location [" + location + "] not resolvable: " + ex.getMessage());
					}
				}
				else {
					throw ex;
				}
			}
		}
	}

比较关键的是@PropertySource 注解中value的值,我们可以通过@PropertySource(classpath:xxx.properties)导入一个properties的配置文件,然后这个路径就是再这里通过value取出,并且可以设置多个
然后对这个路径数组进行循环,并且解析
1.首先始解析占位符,也就是说我们可以在@PropertySource(classpath:${xxx})解析 ${xxx} 这个占位符,这个占位符的解析就需要从spring的环境中去获取,但是注意,这里的这个占位符只能从通过系统变量,或者是jvm的启动变量中获取
2.通过解析之后的路径获取到properties的资源
3.然后将文件的内容进行解析,生成一些键值对,然后加到spring的环境中(environment)
这也就是为什么@Value( ${xxx} )可以获取到propreties里面的一些值,因为在spring 启动的时候,就已经将这些变量解析到environment中了

现在回到doProcessConfigurationClass,前面已经进行了@Component 和 @PropertySource的注解解析,接下来就是@ComponentScan注解的解析了

Set<AnnotationAttributes> componentScans = AnnotationConfigUtils.attributesForRepeatable(
				sourceClass.getMetadata(), ComponentScans.class, ComponentScan.class);
		if (!componentScans.isEmpty() &&
				!this.conditionEvaluator.shouldSkip(sourceClass.getMetadata(), ConfigurationPhase.REGISTER_BEAN)) {
			for (AnnotationAttributes componentScan : componentScans) {

				// The config class is annotated with @ComponentScan -> perform the scan immediately
				// 扫描得到BeanDefinition
				Set<BeanDefinitionHolder> scannedBeanDefinitions =
						this.componentScanParser.parse(componentScan, sourceClass.getMetadata().getClassName());

				// Check the set of scanned definitions for any further config classes and parse recursively if needed
				for (BeanDefinitionHolder holder : scannedBeanDefinitions) {
					BeanDefinition bdCand = holder.getBeanDefinition().getOriginatingBeanDefinition();
					if (bdCand == null) {
						bdCand = holder.getBeanDefinition();
					}

					// 检查扫描所得到BeanDefinition是不是配置Bean,基本上都有@Component注解,所以都是配置类
					if (ConfigurationClassUtils.checkConfigurationClassCandidate(bdCand, this.metadataReaderFactory)) {
						parse(bdCand.getBeanClassName(), holder.getBeanName());
					}
				}
			}
		}

首先是判断这个类有没有加@ComponentScans 或者 @ComponentScan 注解 ,将添加了这些注解的类放入集合,
将这个集合进行遍历,对每个注解进行解析,其实主要就是进行扫描生成对应的BeanDefinition

public Set<BeanDefinitionHolder> parse(AnnotationAttributes componentScan, final String declaringClass) {
		//获取
		ClassPathBeanDefinitionScanner scanner = new ClassPathBeanDefinitionScanner(this.registry,
				componentScan.getBoolean("useDefaultFilters"), this.environment, this.resourceLoader);

		// 设置BeanName生成器
		Class<? extends BeanNameGenerator> generatorClass = componentScan.getClass("nameGenerator");
		boolean useInheritedGenerator = (BeanNameGenerator.class == generatorClass);
		scanner.setBeanNameGenerator(useInheritedGenerator ? this.beanNameGenerator :
				BeanUtils.instantiateClass(generatorClass));

		ScopedProxyMode scopedProxyMode = componentScan.getEnum("scopedProxy");
		if (scopedProxyMode != ScopedProxyMode.DEFAULT) {
			scanner.setScopedProxyMode(scopedProxyMode);
		}
		else {
			Class<? extends ScopeMetadataResolver> resolverClass = componentScan.getClass("scopeResolver");
			scanner.setScopeMetadataResolver(BeanUtils.instantiateClass(resolverClass));
		}

		scanner.setResourcePattern(componentScan.getString("resourcePattern"));

		// 设置IncludeFilter
		for (AnnotationAttributes filter : componentScan.getAnnotationArray("includeFilters")) {
			for (TypeFilter typeFilter : typeFiltersFor(filter)) {
				scanner.addIncludeFilter(typeFilter);
			}
		}

		// 设置ExcludeFilter
		for (AnnotationAttributes filter : componentScan.getAnnotationArray("excludeFilters")) {
			for (TypeFilter typeFilter : typeFiltersFor(filter)) {
				scanner.addExcludeFilter(typeFilter);
			}
		}

		// 设置懒加载  如果通过@ComponentScan 设置的懒加载,那么@ComponetScan扫描出来的类都是懒加载的
		boolean lazyInit = componentScan.getBoolean("lazyInit");
		if (lazyInit) {
			scanner.getBeanDefinitionDefaults().setLazyInit(true);
		}

		Set<String> basePackages = new LinkedHashSet<>();
		//获取路径的信息,然后对路径信息进行解析
		String[] basePackagesArray = componentScan.getStringArray("basePackages");
		for (String pkg : basePackagesArray) {
			String[] tokenized = StringUtils.tokenizeToStringArray(this.environment.resolvePlaceholders(pkg),
					ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);
			Collections.addAll(basePackages, tokenized);
		}
		for (Class<?> clazz : componentScan.getClassArray("basePackageClasses")) {
			basePackages.add(ClassUtils.getPackageName(clazz));
		}
		if (basePackages.isEmpty()) {
			basePackages.add(ClassUtils.getPackageName(declaringClass));
		}

		scanner.addExcludeFilter(new AbstractTypeHierarchyTraversingFilter(false, false) {
			@Override
			protected boolean matchClassName(String className) {
				return declaringClass.equals(className);
			}
		});

		// 开始扫描包路径,得到BeanDefinitionHolder的集合
		return scanner.doScan(StringUtils.toStringArray(basePackages));
	}

其实这个方法最主要做的事情,就是扫描之前的一些设置,例如,需要扫描哪些注解,不需要扫描哪些注解,扫描出来的Bean的beanName如何生成,设置懒加载
然后获取@ComponentScan上面的路径,进过一些处理,就开始进行doScan

protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
		Assert.notEmpty(basePackages, "At least one base package must be specified");
		Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();
		for (String basePackage : basePackages) {
			// 扫描包路径得到BeanDefinition,得到的BeanDefinition是空的,还没有解析类上所定义的注解信息
			Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
			for (BeanDefinition candidate : candidates) {
				// 得到Scope的信息,并设置
				ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
				candidate.setScope(scopeMetadata.getScopeName());
				// 得到beanName
				String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);

				if (candidate instanceof AbstractBeanDefinition) {
					postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
				}
				if (candidate instanceof AnnotatedBeanDefinition) {
					AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
				}
				if (checkCandidate(beanName, candidate)) {
					// 生成BeanDefinitionHolder并注册到registry中
					BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
					definitionHolder =
							AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
					beanDefinitions.add(definitionHolder);
					registerBeanDefinition(definitionHolder, this.registry);
				}
			}
		}
		return beanDefinitions;
	}

其中 findCandidateComponents(basePackage) 就是对路径下的类进行扫描,
扫描出的每一个类都会经过前面设置的 excludeFilters 和 includeFilters ,如果匹配上了 excludeFilters 就会直接return false
如果 匹配上了 includeFilters 才会返回 true
然后通过了过滤的类下扫描出来会封装为一个 ScannedGenericBeanDefinition 对象
完成之后机会生成一个beanName,让后将BeanDefinition的一些默认值设置到 ScannedGenericBeanDefinition

处理完 @ComponentScan 之后就是处理@Import 注解了

private void processImports(ConfigurationClass configClass, SourceClass currentSourceClass,
			Collection<SourceClass> importCandidates, boolean checkForCircularImports) {

		if (importCandidates.isEmpty()) {return;}
		
		if (checkForCircularImports && isChainedImportOnStack(configClass)) {
			this.problemReporter.error(new CircularImportProblem(configClass, this.importStack));
		}
		else {
			this.importStack.push(configClass);
			try {
				for (SourceClass candidate : importCandidates) {
					if (candidate.isAssignable(ImportSelector.class)) {
						// Candidate class is an ImportSelector -> delegate to it to determine imports
						Class<?> candidateClass = candidate.loadClass();
						ImportSelector selector = ParserStrategyUtils.instantiateClass(candidateClass, ImportSelector.class,
								this.environment, this.resourceLoader, this.registry);
						if (selector instanceof DeferredImportSelector) {
							this.deferredImportSelectorHandler.handle(configClass, (DeferredImportSelector) selector);
						}
						else {
							String[] importClassNames = selector.selectImports(currentSourceClass.getMetadata());
							Collection<SourceClass> importSourceClasses = asSourceClasses(importClassNames);
							processImports(configClass, currentSourceClass, importSourceClasses, false);
						}
					}
					else if (candidate.isAssignable(ImportBeanDefinitionRegistrar.class)) {
						// 如果@Import注解中的类实现了ImportBeanDefinitionRegistrar接口,就把该类的实例放入importBeanDefinitionRegistrars中,
						// 后面再执行该实例的registerBeanDefinitions方法
						Class<?> candidateClass = candidate.loadClass();
						ImportBeanDefinitionRegistrar registrar =
								ParserStrategyUtils.instantiateClass(candidateClass, ImportBeanDefinitionRegistrar.class,
										this.environment, this.resourceLoader, this.registry);
						configClass.addImportBeanDefinitionRegistrar(registrar, currentSourceClass.getMetadata());
					}
					else {
						// Candidate class not an ImportSelector or ImportBeanDefinitionRegistrar ->
						// process it as an @Configuration class
						this.importStack.registerImport(
								currentSourceClass.getMetadata(), candidate.getMetadata().getClassName());
						processConfigurationClass(candidate.asConfigClass(configClass));
					}
				}
			}
			catch (BeanDefinitionStoreException ex) {
				........	
			}
			finally {
				this.importStack.pop();
			}
		}
	}

简单介绍一个这个方法的几个参数
ConfigurationClass configClass, SourceClass currentSourceClass, :这两个参数都是配置类的信息
Collection importCandidates :返回当前配置类上 @Import注解中所导入的类的信息
boolean checkForCircularImports : true

接下来关键 的部分就是对这个 importCandidates 进行判断,也就是对Import 注解导入的类进行判断
首先判断是不是ImportSelector.class 的子类,然后判断是不是实现了 ImportBeanDefinitionRegistrar.class,最后就是处理普通的类
1.实现了ImportSelector.class
根据传入的类型,生成一个ImportSelector 对象,其实际类型是传入类型
执行重写的 selectImports 方法,得到String数组(ClassName)
根据这个String数组,加载 出对应的类
进行递归调用,判断导入进来的类是不是实现了ImportSelector 或者是 ImportBeanDefinitionRegistrar
这里也就是说通过@Import导入进来的普通类,一样会当成配置类进行处理
2.实现了ImportBeanDefinitionRegistrar
首先也是实例化了一个ImportBeanDefinitionRegistrar 的对象
但是这里没有调用其重写的方法,而是将这个对象加入当前配置类的一个map属性中
3.普通类型
其实通过@Import 导入进来的类就是一个配置类,所以会将这个配置类添加到configurationClasses这个map中

接下来就是处理@ImportResource

AnnotationAttributes importResource =
				AnnotationConfigUtils.attributesFor(sourceClass.getMetadata(), ImportResource.class);
		if (importResource != null) {
			String[] resources = importResource.getStringArray("locations");
			Class<? extends BeanDefinitionReader> readerClass = importResource.getClass("reader");
			for (String resource : resources) {
				//占位符的填充
				String resolvedResource = this.environment.resolveRequiredPlaceholders(resource);
				configClass.addImportedResource(resolvedResource, readerClass);
			}
		}

对于@ImportResource 而言,就说会获取@ImportResource 上面的值,然后将这个路径添加 到importedResources 这个集合之中,
这个集合就可以理解为:当前配置文件导入资源的路径,这里并不会做解析工作,而是后面进行解析

然后是处理@Bean注解

Set<MethodMetadata> beanMethods = retrieveBeanMethodMetadata(sourceClass);
		for (MethodMetadata methodMetadata : beanMethods) {
			configClass.addBeanMethod(new BeanMethod(methodMetadata, configClass));
		}

这里就会遍历,拿到当前标注了@Bean注解的方法,然后将这些方法封装为BeanMethod,然后添加当配置类的beanMethods 集合中

private void processInterfaces(ConfigurationClass configClass, SourceClass sourceClass) throws IOException {
		for (SourceClass ifc : sourceClass.getInterfaces()) {
			Set<MethodMetadata> beanMethods = retrieveBeanMethodMetadata(ifc);
			for (MethodMetadata methodMetadata : beanMethods) {
				if (!methodMetadata.isAbstract()) {
					// A default method or other concrete method on a Java 8+ interface...
					configClass.addBeanMethod(new BeanMethod(methodMetadata, configClass));
				}
			}
			processInterfaces(configClass, ifc);
		}
	}

这一部分的内容是为了配合java8的功能,去判断配置类实现的接口,因为在java8中,接口可以写一些默认方法,如果接口中有方法加了@Bean一样会扫描出来

if (sourceClass.getMetadata().hasSuperClass()) {
			String superclass = sourceClass.getMetadata().getSuperClassName();
			if (superclass != null && !superclass.startsWith("java") &&
					!this.knownSuperclasses.containsKey(superclass)) {
				this.knownSuperclasses.put(superclass, configClass);
				// Superclass found, return its annotation metadata and recurse
				return sourceClass.getSuperClass();
			}
		}

最后一步就是对父类的处理了,前面提到过的,如果有父类就会将父类添加到sourceClass ,通过do-while循环进行处理

接下来回到processConfigBeanDefinitions 这个方法里面,上述的所有操纵也只是完成了 parser.parse(conadidates)
spring 已经完成了配置类的加载,同时可能得到了跟多的配置类,但是现在需要去解析这些类

private void loadBeanDefinitionsForConfigurationClass(
			ConfigurationClass configClass, TrackedConditionEvaluator trackedConditionEvaluator) {

		if (trackedConditionEvaluator.shouldSkip(configClass)) {
			String beanName = configClass.getBeanName();
			if (StringUtils.hasLength(beanName) && this.registry.containsBeanDefinition(beanName)) {
				this.registry.removeBeanDefinition(beanName);
			}
			this.importRegistry.removeImportingClass(configClass.getMetadata().getClassName());
			return;
		}

		// 当前配置类是不是通过@Import注解导入进来的,如果是则解析该类上面的比如@Scope,@Lazy这些注解信息
		// 然后生成一个AnnotationBeanDefintion 然后进行注册
		if (configClass.isImported()) {
			registerBeanDefinitionForImportedConfigurationClass(configClass);
		}
		//处理@Bean 生成 BeanDefintion 进行注册
		//这里生成的BeanDefinition 是ConfigurationClassBeanDefinition
		//需要注意的几个点 @Bean 的注入模型是AUTOWIRE_CONSTRUCTOR,也就是通过构造方法
		for (BeanMethod beanMethod : configClass.getBeanMethods()) {
			loadBeanDefinitionsForBeanMethod(beanMethod);
		}

		// 从配置类中Import进来的资源文件中加载BeanDefinition,比如xml文件
		loadBeanDefinitionsFromImportedResources(configClass.getImportedResources());
		// 调用ImportBeanDefinitionRegistrar对象的registerBeanDefinitions方法加载BeanDefinition
		loadBeanDefinitionsFromRegistrars(configClass.getImportBeanDefinitionRegistrars());
	}

进过这个方法之后,spring就算是完成了扫描和BeanDefinition的注册了