Spring 5.x 源码之旅-3BeanDefinition实现与ClassPathBeanDefinitionScanner

本文介绍了从深入多线程、设计模式到JVM源码的深入学习路径,以及各种JavaBean定义(如AbstractBeanDefinition、GenericBeanDefinition、AnnotatedBeanDefinition等)的原理和实现。同时提及了ClassPathBeanDefinitionScanner用于扫描注解并配置Spring应用的工具。

作者简介:大家好,我是smart哥,前中兴通讯、美团架构师,现某互联网公司CTO

联系qq:184480602,加我进群,大家一起学习,一起进步,一起对抗互联网寒冬

学习必须往深处挖,挖的越深,基础越扎实!

阶段1、深入多线程

阶段2、深入多线程设计模式

阶段3、深入juc源码解析


阶段4、深入jdk其余源码解析


阶段5、深入jvm源码解析

码哥源码部分

码哥讲源码-原理源码篇【2024年最新大厂关于线程池使用的场景题】

码哥讲源码【炸雷啦!炸雷啦!黄光头他终于跑路啦!】

码哥讲源码-【jvm课程前置知识及c/c++调试环境搭建】

​​​​​​码哥讲源码-原理源码篇【揭秘join方法的唤醒本质上决定于jvm的底层析构函数】

码哥源码-原理源码篇【Doug Lea为什么要将成员变量赋值给局部变量后再操作?】

码哥讲源码【你水不是你的错,但是你胡说八道就是你不对了!】

码哥讲源码【谁再说Spring不支持多线程事务,你给我抽他!】

终结B站没人能讲清楚红黑树的历史,不服等你来踢馆!

打脸系列【020-3小时讲解MESI协议和volatile之间的关系,那些将x86下的验证结果当作最终结果的水货们请闭嘴】

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,后面马上会用到他,他就是来解析我们配置类的。而扫描器不是他内部用的,是给我们用户用的,内部他要用还会创建一个。我们后面就继续看他怎么处理我们的配置类。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值