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.ManagedBean
和javax.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
,用来做字节码文件元数据的缓存,创建了CandidateComponentsIndex
,spring
内部定义的组件,在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
,后面马上会用到他,他就是来解析我们配置类的。而扫描器不是他内部用的,是给我们用户用的,内部他要用还会创建一个。我们后面就继续看他怎么处理我们的配置类。
好了,今天就到这里了,希望对学习理解有帮助,大神看见勿喷,仅为自己的学习理解,能力有限,请多包涵。
Java 面试宝典是大明哥全力打造的 Java 精品面试题,它是一份靠谱、强大、详细、经典的 Java 后端面试宝典。它不仅仅只是一道道面试题,而是一套完整的 Java 知识体系,一套你 Java 知识点的扫盲贴。
它的内容包括:
- 大厂真题:Java 面试宝典里面的题目都是最近几年的高频的大厂面试真题。
- 原创内容:Java 面试宝典内容全部都是大明哥原创,内容全面且通俗易懂,回答部分可以直接作为面试回答内容。
- 持续更新:一次购买,永久有效。大明哥会持续更新 3+ 年,累计更新 1000+,宝典会不断迭代更新,保证最新、最全面。
- 覆盖全面:本宝典累计更新 1000+,从 Java 入门到 Java 架构的高频面试题,实现 360° 全覆盖。
- 不止面试:内容包含面试题解析、内容详解、知识扩展,它不仅仅只是一份面试题,更是一套完整的 Java 知识体系。
- 宝典详情:https://www.yuque.com/chenssy/sike-java/xvlo920axlp7sf4k
- 宝典总览:https://www.yuque.com/chenssy/sike-java/yogsehzntzgp4ly1
- 宝典进展:https://www.yuque.com/chenssy/sike-java/en9ned7loo47z5aw
目前 Java 面试宝典累计更新 400+ 道,总字数 42w+。大明哥还在持续更新中,下图是大明哥在 2024-12 月份的更新情况:
想了解详情的小伙伴,扫描下面二维码加大明哥微信【daming091】咨询
同时,大明哥也整理一套目前市面最常见的热点面试题。微信搜[大明哥聊 Java]或扫描下方二维码关注大明哥的原创公众号[大明哥聊 Java] ,回复【面试题】 即可免费领取。