为什么说Spring框架只会将标注有@Component、@Named、@ManagedBean注解的类加入到容器? @Configuration、@Controller又为什么会加入到容器中?

Spring只会把标注有@Component、@Named、@ManagedBean注解的类加入到容器内,但是我们使用@Controller、@Service、@Repository注解也能加入到容器,这是怎么实现的?

Spring注解配置启动的整个过程是这样的,首先Spring遍历所有文件夹获取到类文件,将类文件封装成Resource,使用MetadataReader对Resource进行读取,读取到类的注解等信息,然后再根据注解进行筛选,符合条件的封装成BeanDefinition。有了BeanDefinition就可以生成Bean了。在这个过程中负责扫描的是ClassPathScanningCandidateComponentProvider类,来看一下这个类的实现

public class ClassPathScanningCandidateComponentProvider2 implements EnvironmentCapable, ResourceLoaderAware {

	//...
	private final List<TypeFilter> includeFilters = new LinkedList<>();
	//...

	//在类实例化是调用了这个方法,注册了@Component、@Named、@ManagedBean三个注解
	protected void registerDefaultFilters() {
		//注册@Component
		this.includeFilters.add(new AnnotationTypeFilter(Component.class));
		ClassLoader cl = ClassPathScanningCandidateComponentProvider2.class.getClassLoader();
		try {
			//注册@ManagedBean
			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 {
			//注册@Named
			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.
		}
	}
	//...
	
	//这个是扫描的方法,这里对类上的注解进行了过滤,只保留了标有@Component、@Named、@ManagedBean的类,并封装成BeanDefinition
	private Set<BeanDefinition> scanCandidateComponents(String basePackage) {
		Set<BeanDefinition> candidates = new LinkedHashSet<>();
		//...
		//使用资源加载器,获取所有类文件,并分装成Resource
		Resource[] resources = getResourcePatternResolver().getResources(packageSearchPath);
		//...
		for (Resource resource : resources) {
			//...
			//读取类的信息
			MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource);
			//这个判断是关键,只有标注有@Component、@ManagedBean、@Named注解的才会是true,走下面流程生成BeanDefinition
			if (isCandidateComponent(metadataReader)) {
				//生成Beand定义
				ScannedGenericBeanDefinition sbd = new ScannedGenericBeanDefinition(metadataReader);
				sbd.setSource(resource);
				//...
				//添加到结果集中
				candidates.add(sbd);
			}
			//...
		}
		return candidates;
	}
	//...

	protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException {
		//...
		//这里遍历includeFilters,在类registerDefaultFilters方法执行后,向includeFilters添加了@Component、@Named、@ManagedBean类型过滤器
		for (TypeFilter tf : this.includeFilters) {
			//判断类是否标注有tf类型的注解
			if (tf.match(metadataReader, getMetadataReaderFactory())) {
				//Condition条件判断
				return isConditionMatch(metadataReader);
			}
		}
		return false;
	}
}

总结一下,首先在类实例化的时候调用了registerDefaultFilters方法注册了@Component、@Named、@ManagedBean类型的过滤器。在扫描完成后调用isCandidateComponent方法逐个判断,符合条的才会生成BeanDefinition。isCandidateComponent方法内遍历注册的过滤器与类进行匹配,符合的再进行Condition判断成功的返回true。

看完上面代码好像Spring只能识别三个注解,那@Controller、@Service、@Repository又是怎么识别的呢?请看下篇

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值