Spring IOC基于注解启动示例详析

网友投稿 221 2023-07-08

Spring IOC基于注解启动示例详析

Spring 基于注解启动

主要有两个Class实现注解启动

AnnotationConfigApplicationContext

AnnotationConfigWebApplicationContext

我们以AnnotationConfigApplicationContext 为研究对象

引入Spring 最小依赖

org.springframework

spring-context

${spring.version}

编写器启动代码

public static void main(String[] args) {

AnnotationConfigApplicationContext applicationContext = new AnnotationConfigApplicationContext();

applicationContext.register(BeanConfig.class);

applicationContext.refresh();

Date date = applicationContext.getBean("date",Date.class);

System.out.println(date);

}

AnnotationConfigApplicationContext 构造函数

public AnnotationConfigApplicationContext() {

//负责注册Class ,读取器

this.reader = new AnnotatedBeanDefinitionReader(this);

//负责扫描指定类路径下的Class,注册bean

this.scanner = new ClassPathBeanDefinitionScanner(this);

}

AnnotatedBeanDefinitionReader 构造方法

public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry) {

this(registry, getOrCreateEnvironment(registry));

}

public AnnotatedBeanDefinitionReader(BeanDefinitionRegistry registry, Environment environment) {

Assert.notNull(registry, "BeanDefinitionRegistry must not be null");

Assert.notNull(environment, "Environment must not be null");

this.registry = registry;

//初始化ConditionEvaluator

this.conditionEvaluator = new ConditionEvaluator(registry, environment, null);

/** 在給定的注册表中註冊所有相关的post processors

* 判断容器是否已经存在给定注册表的bean,如果没有注册bean,并将bean放入容器中

* 把所有的处理处理器列出来

* ConfigurationClassPostProcessor 內部管理的配置注解处理器

* AutowiredAnnotationBeanPostProcessor 内部管理@Autowired 的处理器

* RequiredAnnotationBeanPostProcessor @Required的处理器

* CommonAnnotationBeanPostProcessor jsR-250注解处理器 ,先判断是否支持jsr,如果支持注册

* PersistenceAnnotationBeanPostProcessor JPA管理 先使用类加载器查找是否存在,如果有这个包则注册

* EventListenerMethodProcessor @EventListener的处理器

* DefaultEventListenerFactory 管理EventListenerFactory处理器

*/

AnnotationConfigUtils.registerAnnotationConfigProcessors(this.registry);

}

ConditionEvaluator 这个对象干什么,点击进去

public ConditionEvaluator(@Nullable BeanDefinitionRegistry registry,

@Nullable Environment environment, @Nullable ResourceLoader resourceLoader) {

this.context = new ConditionContextImpl(registry, environment, resourceLoader);

}

//ConditionContextImpl 实现了ConditionContext接口,ConditionEvaluator静态内部类

public ConditionContextImpl(@Nullable BeanDefinitionRegistry registry,

@Nullable Environment environment, @Nullable ResourceLoader resourceLoader) {

this.registry = registry;

this.beanFactory = deduceBeanFactory(registry);

this.environment = (environment != null ? environment : deduceEnvironment(registry));

this.resourceLoader = (resourceLoader != null ? resourceLoader : deduceResourceLoader(registry));

this.classLoader = deduceClassLoader(resourceLoader, this.beanFactory);

}

可以知道ConditionEvaluator使用外部传参的方法初始化了Spring容器顶级对象

BeanFactory,Environment,ResourceLoader,ClassLoader。在将这些传给ConditionContextImpl为接下来的解析@Conditional注解做好准备

ClassPathBeanDefinitionScanner构造函数

public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry) {

this(registry, true);

}

public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters) {

this(registry, useDefaultFilters, getOrCreateEnvironment(registry));

}

public ClassPathBeanDefinitionScanner(BeanDefinitionRegistry registry, boolean useDefaultFilters,

Environment environment) {

this(registry, useDefaultFilters, environment,

(registry instanceof ResourceLoader ? (ResourceLoader) registry : null));

}

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);

}

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.debug("JSR-250 'javax.annotation.ManagedBean' found and supported for component scanning");

}

catch (ClassNotFoundException ex) {

}

try {

this.includeFilters.add(new AnnotationTypeFilter(

((Class extends Annotation>) ClassUtils.forName("javax.inject.Named", cl)), false));

logger.debug("JSR-330 'javax.inject.Named' annotation found and supported for component scanning");

}

catch (ClassNotFoundException ex) {

// JSR-330 API not available - simply skip.

}

}

绕了地球几圈了,其实就是将Spring 顶级接口 Environment,ResourceLoader赋值,使用默认注解过滤器,首先将@Component加入List中,判断当前环境是否支持JSR-250,JSR-330,相应加入过滤器中。也就是这个扫描器默认只扫描@Component或者JSR-250,JSR-330的标记的Class。

applicationContext.register(BeanConfig.class)

public void register(Class>... annotatedClasses) {

Assert.notEmpty(annotatedClasses, "At least one annotated class must be specified");

this.reader.register(annotatedClasses); //调用 刚刚初始化读取器

}

|

============================AnnotatedBeanDefinitionReader 读取器代码======================================================================================================

public void register(Class>... annotatedClasses) {

for (Class> annotatedClass : annotatedClasses) {

registerBean(annotatedClass);

}

}

public void registerBean(Class> annotatedClass) {

doRegisterBean(annotatedClass, null, null, null);

}

/**

*从给定的bean解析Class给定的注解,执行相应的初始化,保存到Spring容器中

*/

void doRegisterBean(Class annotatedClass, @Nullable Supplier instanceSupplier, @Nullable String name,

@Nullable Class extends Annotation>[] qualifiers, BeanDefinitionCustomizer... definitionCustomizers) {

//根据Class的Annotated 得出元数据 AnnotationMetadata

AnnotatedGenericBeanDefinition abd = new AnnotatedGenericBeanDefinition(annotatedClass);

/**

* 判断注册的Class 是否包含@Conditional注解,如果有获取全部value,放入List中

* 排序后,遍历所有的Conditiion的实现,使用反射获取对象,执行matches方法,

* 如果发现有返回false,中断循环直接返回true,

*/

if (this.conditionEvaluator.shouldSkip(abd.getMetadata())) { //如果 @Conditional条件不满足,不进行注册

return;

}

abd.setInstanceSupplier(instanceSupplier);

//解析Class是否有@Scope,解析@Scope注解返回ScopeMetadata对象,没有直接返回空

ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(abd);

abd.setScope(scopeMetadata.getScopeName());

//判断注解上Value是否有值,有就使用这个作为BeanName,没有则取类名

String beanName = (name != null ? name : this.beanNameGenerator.generateBeanName(abd, this.registry));

//继续解析AnnotationMetadata的@Lazy,@Primary,@DependsOn,@Role,@Description的注解,放入结果放入对象的属性中

AnnotationConfigUtils.processCommonDefinitionAnnotations(abd);

//这个类只是BeanDefinition 包装类

BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(abd, beanName);

//是否需要代理类,如果是则修改内部属性,重新生成BeanDefinition 对象

definitionHolder = AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);

//调用DefaultListableBeanFactory.registerBeanDefinition的方法,做一些安全性校验再,将definitionHolder 放入register容器中

BeanDefinitionReaderUtils.registerBeanDefinition(definitionHolder, this.registry);

}

这个方法就是将注册的Bean,解析Class上的注解,初始化注解数据,做相应处理,转化成BeanDefinition ,放入Spring 容器中保存起来。

我们看下BeanDefinition是怎么实现注册到Spring的容器中,主要由DefaultListableBeanFactory.registerBeanDefinition来实现

将beanDefinition注册到Spring容器中,并没有太多复杂的逻辑,只是做一些安全性的检查。

BeanDefinition

一个BeanDefinition描述了一个bean的实例,包括属性值,构造方法参数值和继承自它的类的更多信息。BeanDefinition仅仅是一个最简单的接口,主要功能是允许BeanFactoryPostProcessor 例如PropertyPlaceHolderConfigure 能够检索并修改属性值和别的bean的元数据(译注)

Spring 容器beanDefinition主要分为RootBeanDefinition,AnnotatedGenericBeanDefinition这两种

RootBeanDefinition     Spring Factory中的特定bean

AnnotatedGenericBeanDefinition      用户自定义bean

Spring 启动流程总结

AnnotationConfigApplicationContext 初始化.png

这些BeanDefinition只是放入到Spirng 容器中,并没有进行任何初始化对象的操作,真正的IOC操作都在refresh(),这个方法有空再进行分析。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

版权声明:本文内容由网络用户投稿,版权归原作者所有,本站不拥有其著作权,亦不承担相应法律责任。如果您发现本站中有涉嫌抄袭或描述失实的内容,请联系我们jiasou666@gmail.com 处理,核实后本网站将在24小时内删除侵权内容。

上一篇:Spring boot定时任务的原理及动态创建详解
下一篇:java编程题之合并两个排序的链表
相关文章

 发表评论

暂时没有评论,来抢沙发吧~