03、Spring5.x源码之旅三BeanDefinition实现与ClassPathBeanDefinitionScanner

  • AbstractBeanDefinition抽象实现
    • GenericBeanDefinition标准的bean定义
  • AnnotatedBeanDefinition注解bean定义
    • AnnotatedGenericBeanDefinition注解通用bean定义
  • RootBeanDefinition合并bean定义
  • ClassPathBeanDefinitionScanner扫描类定义
    • registerDefaultFilters注册默认过滤器
  • setResourceLoader设置资源加载器

AbstractBeanDefinition抽象实现

我们继续上一篇的bean定义说明。
*
其实就是继承了上面的bean元数据访问器并且实现了bean定义接口,这样的话就可以让bean的元数据和bean的定义接口联系起来了,也就是数据和操作结合了。这个类比较长,我不打算从头到位贴上来,说几个要注意的地方:

	//this.beanClasss是对象,可以是String也可以是Class
	@Override
	@Nullable
	public String getBeanClassName() {
   
     
		Object beanClassObject = this.beanClass;
		if (beanClassObject instanceof Class) {
   
     
			return ((Class<?>) beanClassObject).getName();
		}
		else {
   
     
			return (String) beanClassObject;
		}
	}
	
	//可以传Class
	public void setBeanClass(@Nullable Class<?> beanClass) {
   
     
		this.beanClass = beanClass;
	}
	//判断beanClass是否是个Class对象
	public boolean hasBeanClass() {
   
     
		return (this.beanClass instanceof Class);
	}

	//定义了一个克隆的抽象方法
	public abstract AbstractBeanDefinition cloneBeanDefinition();

GenericBeanDefinition标准的bean定义

*
增加了一个父类名字,实现了抽象克隆方法:

	@Nullable
	private String parentName;

	@Override
	public AbstractBeanDefinition cloneBeanDefinition() {
   
     
		return new GenericBeanDefinition(this);
	}

AnnotatedBeanDefinition注解bean定义

*
加了两个新方法,其实就是注解的元数据和工厂方法的元数据,这些数据在进行解析处理的时候需要用到。

public interface AnnotatedBeanDefinition extends BeanDefinition {
   
     

	
	AnnotationMetadata getMetadata();

	
	@Nullable
	MethodMetadata getFactoryMethodMetadata();

}

AnnotatedGenericBeanDefinition注解通用bean定义

*
别看他的继承如图那么复杂,其实就是上面讲的这些,就是将通用bean定义GenericBeanDefinition和注解bean定义AnnotatedBeanDefinition拼起来了。

public class AnnotatedGenericBeanDefinition extends GenericBeanDefinition implements AnnotatedBeanDefinition {
   
     
	//注解元数据
	private final AnnotationMetadata metadata;
	//工厂方法源数据
	@Nullable
	private MethodMetadata factoryMethodMetadata;
	//通过beanClass来获取元数据
	public AnnotatedGenericBeanDefinition(Class<?> beanClass) {
   
     
		setBeanClass(beanClass);
		this.metadata = AnnotationMetadata.introspect(beanClass);
	}

	//通过注解元数据来获取beanClass
	public AnnotatedGenericBeanDefinition(AnnotationMetadata metadata) {
   
     
		Assert.notNull(metadata, "AnnotationMetadata must not be null");
		if (metadata instanceof StandardAnnotationMetadata) {
   
     
			setBeanClass(((StandardAnnotationMetadata) metadata).getIntrospectedClass());
		}
		else {
   
     
			setBeanClassName(metadata.getClassName());
		}
		this.metadata = metadata;
	}

	//通过注解元数据和方法元数据创建
	public AnnotatedGenericBeanDefinition(AnnotationMetadata metadata, MethodMetadata factoryMethodMetadata) {
   
     
		this(metadata);
		Assert.notNull(factoryMethodMetadata, "MethodMetadata must not be null");
		setFactoryMethodName(factoryMethodMetadata.getMethodName());
		this.factoryMethodMetadata = factoryMethodMetadata;
	}
	@Override
	public final AnnotationMetadata getMetadata() {
   
     
		return this.metadata;
	}

	@Override
	@Nullable
	public final MethodMetadata getFactoryMethodMetadata() {
   
     
		return this.factoryMethodMetadata;
	}

}

RootBeanDefinition合并bean定义

*
这个主要是将很多其他类型的BeanDefinition在运行时合并起来,具体的用法后面会说,现在只要知道基本内部的定义都是用这个的。

ClassPathBeanDefinitionScanner扫描类定义

这个就是来扫描我们注解,比如@Component,@Repository,@Service,@Controller,或者java扩展的javax.annotation.ManagedBeanjavax.inject.Named,当然创建的时候他没有干什么事,只是做了一些基本的配置,比如注册过滤器,比如能过滤@Component注解,然后设置ResourceLoader,创建元数据缓存CachingMetadataReaderFactory等。

	public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters,
			Environment environment, @Nullable ResourceLoader resourceLoader) {
   
     

		Assert.notNull(registry, "BeanDefinitionRegistry must not be null");
		this.registry = registry;
		//是否使用默认过滤器
		if (useDefaultFilters) {
   
     
			registerDefaultFilters();
		}
		setEnvironment(environment);
		setResourceLoader(resourceLoader);
	}

registerDefaultFilters注册默认过滤器

主要是@Component,但是还有其他的一些java扩展的。

protected void registerDefaultFilters() {
   
     
		this.includeFilters.add(new AnnotationTypeFilter(Component.class));
		ClassLoader cl = ClassPathScanningCandidateComponentProvider.class.getClassLoader();
		try {
   
     
			this.includeFilters.add(new AnnotationTypeFilter(
					((Class<? extends Annotation>) ClassUtils.forName("javax.annotation.ManagedBean", cl)), false));
			logger.trace("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning");
		}
		catch (ClassNotFoundException ex) {
   
     
			// JSR-250 1.1 API (as included in Java EE 6) not available - simply skip.
		}
		try {
   
     
			this.includeFilters.add(new AnnotationTypeFilter(
					((Class<? extends Annotation>) ClassUtils.forName("javax.inject.Named", cl)), false));
			logger.trace("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");
		}
		catch (ClassNotFoundException ex) {
   
     
			// JSR-330 API not available - simply skip.
		}
	}

setResourceLoader设置资源加载器

这里创建了ResourcePatternResolver,用来解析URL资源,创建了CachingMetadataReaderFactory,用来做字节码文件元数据的缓存,创建了CandidateComponentsIndexspring内部定义的组件,在META-INF/spring.components里,貌似现在还没用到。

	@Override
	public void setResourceLoader(@Nullable ResourceLoader resourceLoader) {
   
     
		this.resourcePatternResolver = ResourcePatternUtils.getResourcePatternResolver(resourceLoader);
		this.metadataReaderFactory = new CachingMetadataReaderFactory(resourceLoader);
		this.componentsIndex = CandidateComponentsIndexLoader.loadIndex(this.resourcePatternResolver.getClassLoader());
	}

至此AnnotationConfigApplicationContext创建完成了,主要创建了读取器和扫描器,读取器里面又注册了注解后置处理器的bean定义,特别是这个ConfigurationClassPostProcessor,后面马上会用到他,他就是来解析我们配置类的。而扫描器不是他内部用的,是给我们用户用的,内部他要用还会创建一个。我们后面就继续看他怎么处理我们的配置类。

好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。