springboot扫描组件_SpringBoot2.x基础篇:带你了解扫描Package自动注册Bean

知识改变命运,撸码使我快乐,2020继续游走在开源界

点赞再看,养成习惯

给我来个Star吧,点击了解下基于SpringBoot的组件化接口服务落地解决方案

我们一直在使用SpringBoot来开发应用程序,但是为什么在项目启动时就会自动注册使用注解@Component、@Service、@RestController...标注的Bean呢?

推荐阅读

默认扫描目录

SpringBoot把入口类所在的Package作为了默认的扫描目录,这也是一个约束,如果我们把需要被注册到IOC的类创建在扫描目录下就可以实现自动注册,否则则不会被注册。

如果你入口类叫做ExampleApplication,它位于org.minbox.chapter目录下,当我们启动应用程序时就会自动扫描org.minbox.chapter同级目录、子级目录下全部注解的类,如下所示:

. src/main/java

├── org.minbox.chapter

│ ├── ExampleApplication.java

│ ├── HelloController.java

│ ├── HelloExample.java

│ └── index

│ │ └── IndexController.java

├── com.hengboy

│ ├── TestController.java

└──

HelloController.java、HelloExample.java与入口类ExampleApplication.java在同一级目录下,所以在项目启动时可以被扫描到。

IndexController.java则是位于入口类的下级目录org.minbox.chapter.index内,因为支持下级目录扫描,所以它也可以被扫描到。

TestController.java位于com.hengboy目录下,默认无法扫描到。

自定义扫描目录

在上面目录结构中位于com.hengboy目录下的TestController.java类,默认情况下是无法被扫描并注册到IOC容器内的,如果想要扫描该目录下的类,下面有两种方法。

方法一:使用@ComponentScan注解

@ComponentScan({"org.minbox.chapter", "com.hengboy"})

方法二:使用scanBasePackages属性

@SpringBootApplication(scanBasePackages = {"org.minbox.chapter", "com.hengboy"})

注意事项:配置自定义扫描目录后,会覆盖掉默认的扫描目录,如果你还需要扫描默认目录,那么你要进行配置扫描目录,在上面自定义配置中,如果仅配置扫描com.hengboy目录,则org.minbox.chapter目录就不会被扫描。

追踪源码

下面我们来看下SpringBoot源码是怎么实现自动化扫描目录下的Bean,并将Bean注册到容器内的过程。

由于注册的流程比较复杂,挑选出具有代表性的流程步骤来进行讲解。

获取BasePackages

在org.springframework.context.annotation.ComponentScanAnnotationParser#parse方法内有着获取basePackages的业务逻辑,源码如下所示:

SetbasePackages = new LinkedHashSet<>();

// 获取@ComponentScan注解配置的basePackages属性值

String[] basePackagesArray = componentScan.getStringArray("basePackages");

// 将basePackages属性值加入Set集合内

for (String pkg : basePackagesArray) {

String[] tokenized = StringUtils.tokenizeToStringArray(this.environment.resolvePlaceholders(pkg),

ConfigurableApplicationContext.CONFIG_LOCATION_DELIMITERS);

Collections.addAll(basePackages, tokenized);

}

// 获取@ComponentScan注解的basePackageClasses属性值

for (Class> clazz : componentScan.getClassArray("basePackageClasses")) {

// 获取basePackageClasses所在的package并加入Set集合内

basePackages.add(ClassUtils.getPackageName(clazz));

}

// 如果并没有配置@ComponentScan的basePackages、basePackageClasses属性值

if (basePackages.isEmpty()) {

// 使用Application入口类的package作为basePackage

basePackages.add(ClassUtils.getPackageName(declaringClass));

}

获取basePackages分为了那么三个步骤,分别是:

获取@ComponentScan注解basePackages属性值

获取@ComponentScan注解basePackageClasses属性值

将Application入口类所在的package作为默认的basePackages

注意事项:根据源码也就证实了,为什么我们配置了basePackages、basePackageClasses后会把默认值覆盖掉,这里其实也不算是覆盖,是根本不会去获取Application入口类的package。

扫描Packages下的Bean

获取到全部的Packages后,通过org.springframework.context.annotation.ClassPathBeanDefinitionScanner#doScan方法来扫描每一个Package下使用注册注解(@Component、@Service、@RestController...)标注的类,源码如下所示:

protected SetdoScan(String... basePackages) {

// 当basePackages为空时抛出IllegalArgumentException异常

Assert.notEmpty(basePackages, "At least one base package must be specified");

SetbeanDefinitions = new LinkedHashSet<>();

// 遍历每一个basePackage,扫描package下的全部Bean

for (String basePackage : basePackages) {

// 获取扫描到的全部Bean

Setcandidates = findCandidateComponents(basePackage);

// 遍历每一个Bean进行处理注册相关事宜

for (BeanDefinition candidate : candidates) {

// 获取作用域的元数据

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

candidate.setScope(scopeMetadata.getScopeName());

// 获取Bean的Name

String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);

if (candidate instanceof AbstractBeanDefinition) {

postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);

}

// 如果是注解方式注册的Bean

if (candidate instanceof AnnotatedBeanDefinition) {

// 处理Bean上的注解属性,相应的设置到BeanDefinition(AnnotatedBeanDefinition)类内字段

AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);

}

// 检查是否满足注册的条件

if (checkCandidate(beanName, candidate)) {

// 声明Bean具备的基本属性

BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);

// 应用作用域代理模式

definitionHolder =

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

// 写入返回的集合

beanDefinitions.add(definitionHolder);

// 注册Bean

registerBeanDefinition(definitionHolder, this.registry);

}

}

}

return beanDefinitions;

}

在上面源码中会扫描每一个basePackage下通过注解定义的Bean,获取Bean注册定义对象后并设置一些基本属性。

注册Bean

扫描到basePackage下的Bean后会直接通过org.springframework.beans.factory.support.BeanDefinitionReaderUtils#registerBeanDefinition方法进行注册,源码如下所示:

public static void registerBeanDefinition(

BeanDefinitionHolder definitionHolder, BeanDefinitionRegistry registry)

throws BeanDefinitionStoreException {

// 注册Bean的唯一名称

String beanName = definitionHolder.getBeanName();

// 通过BeanDefinitionRegistry注册器进行注册Bean

registry.registerBeanDefinition(beanName, definitionHolder.getBeanDefinition());

// 如果存在别名,进行注册Bean的别名

String[] aliases = definitionHolder.getAliases();

if (aliases != null) {

for (String alias : aliases) {

registry.registerAlias(beanName, alias);

}

}

}

通过org.springframework.beans.factory.support.BeanDefinitionRegistry#registerBeanDefinition注册器内的方法可以直接将Bean注册到IOC容器内,而BeanName则是它生命周期内的唯一名称。

总结

通过本文的讲解我想你应该已经了解了SpringBoot应用程序启动时为什么会自动扫描package并将Bean注册到IOC容器内,虽然项目启动时间很短暂,不过这是一个非常复杂的过程,在学习过程中大家可以使用Debug模式来查看每一个步骤的逻辑处理。

作者个人 博客

使用开源框架 ApiBoot 助你成为Api接口服务架构师

文章来源: www.oschina.net,作者:恒宇少年,版权归原作者所有,如需转载,请联系作者。

原文链接:https://my.oschina.net/yuqiyu/blog/3173957

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值