• 【Spring注解必知必会】深度解析@Component注解实现原理


    概述

    想必@Component注解大家一直在使用,只要类上加上它,就可以被Spring容器管理,那大家有想过它是怎么实现的吗?本篇文章就带领到家揭秘。

    注解介绍

    用来标记的类是一个“组件”或者说是一个Bean,Spring会自动扫描标记@Component注解的类作为一个Spring Bean对象。

    注解源码:

    1. @Target(ElementType.TYPE)
    2. @Retention(RetentionPolicy.RUNTIME)
    3. @Documented
    4. @Indexed
    5. public @interface Component {
    6. /**
    7. * The value may indicate a suggestion for a logical component name,
    8. * to be turned into a Spring bean in case of an autodetected component.
    9. * @return the suggested component name, if any (or empty String otherwise)
    10. */
    11. String value() default "";
    12. }
    13. 复制代码

    属性说明:

    • value: 自定义当前组件或者说bean的名称,可以不配置, 不配置的话默认为组件的首字母小写的类名。

    元注解说明:

    • 该注解只能使用在类,接口、枚举、其他注解上
    • 该注解的生命周期是运行时JVM
    • @Indexed元注解在spring 5.0引入,用于项目编译打包时,会在自动生成META-INF/spring.components文件,简历索引,从而提高组件扫描效率,减少应用启动时间。

    注解使用

    1. 定义Person类,被@Component注解修饰

    1. 检查Person类是否在扫描路径下

    1. 获取bean验证

    小结: 通过添加@Component能够将类转为Spring中的Bean对象,前提是能过够被扫描到。

    原理解析

    阅读源码,我们查看@Component注解的源码,从中可以看到一个关键的类ClassPathBeanDefinitionScanner,我们可以从这个类下手,找到切入点。

    分析ClassPathBeanDefinitionScanner类,找到核心方法doscan, 打个断点,了解整个调用链路。

    具体分析结果如下:

    1. SpringBoot应用启动会注册ConfigurationClassPostProcessor这个Bean,它实现了BeanDefinitionRegistryPostProcessor接口,而这个接口是Spring提供的一个扩展点,可以往BeanDefinition Registry中添加BeanDefintion。所以,只要能够扫描到@Component注解的类,并且把它注册到BeanDefinition Registry中即可。

    1. 关键方法ConfigurationClassPostProcessorpostProcessBeanDefinitionRegistry,查找@Component的类,并进行注册。

    1. 我们直接跳到是如何查找@Component的类的,核心方法就是ClassPathBeanDefinitionScanner#doScan
    1. protected Set<BeanDefinitionHolder> doScan(String... basePackages) {
    2. Assert.notEmpty(basePackages, "At least one base package must be specified");
    3. Set<BeanDefinitionHolder> beanDefinitions = new LinkedHashSet<>();
    4. // 遍历多个扫描目录,如本例中的com.alvinlkk
    5. for (String basePackage : basePackages) {
    6. // 核心方法查找所有符合条件的BeanDefinition, 该方法后面重点关注
    7. Set<BeanDefinition> candidates = findCandidateComponents(basePackage);
    8. // 遍历找到的BeanDefinition
    9. for (BeanDefinition candidate : candidates) {
    10. ScopeMetadata scopeMetadata = this.scopeMetadataResolver.resolveScopeMetadata(candidate);
    11. candidate.setScope(scopeMetadata.getScopeName());
    12. String beanName = this.beanNameGenerator.generateBeanName(candidate, this.registry);
    13. if (candidate instanceof AbstractBeanDefinition) {
    14. postProcessBeanDefinition((AbstractBeanDefinition) candidate, beanName);
    15. }
    16. if (candidate instanceof AnnotatedBeanDefinition) {
    17. AnnotationConfigUtils.processCommonDefinitionAnnotations((AnnotatedBeanDefinition) candidate);
    18. }
    19. // 验证BeanDefinition
    20. if (checkCandidate(beanName, candidate)) {
    21. BeanDefinitionHolder definitionHolder = new BeanDefinitionHolder(candidate, beanName);
    22. definitionHolder =
    23. AnnotationConfigUtils.applyScopedProxyMode(scopeMetadata, definitionHolder, this.registry);
    24. beanDefinitions.add(definitionHolder);
    25. // 注册BeanDefinition到registry中
    26. registerBeanDefinition(definitionHolder, this.registry);
    27. }
    28. }
    29. }
    30. return beanDefinitions;
    31. }
    32. 复制代码
    1. 重点关注ClassPathBeanDefinitionScanner#findCandidateComponents方法,找出候选的Bean Component。
    1. public Set findCandidateComponents(String basePackage) {
    2. // 判断组件是否加了索引,打包后默认会有索引,用于加快扫描
    3. if (this.componentsIndex != null && indexSupportsIncludeFilters()) {
    4. return addCandidateComponentsFromIndex(this.componentsIndex, basePackage);
    5. }
    6. // 重点查看else逻辑
    7. else {
    8. return scanCandidateComponents(basePackage);
    9. }
    10. }
    11. 复制代码
    1. private Set<BeanDefinition> scanCandidateComponents(String basePackage) {
    2. Set<BeanDefinition> candidates = new LinkedHashSet<>();
    3. try {
    4. // 解析出需要扫描的路径,本例是classpath*:com/alvinlkk/**/*.class
    5. String packageSearchPath = ResourcePatternResolver.CLASSPATH_ALL_URL_PREFIX +
    6. resolveBasePackage(basePackage) + '/' + this.resourcePattern;
    7. // 根据扫描路径找到所有的Resource
    8. Resource[] resources = getResourcePatternResolver().getResources(packageSearchPath);
    9. boolean traceEnabled = logger.isTraceEnabled();
    10. boolean debugEnabled = logger.isDebugEnabled();
    11. // 遍历扫描路径
    12. for (Resource resource : resources) {
    13. if (traceEnabled) {
    14. logger.trace("Scanning " + resource);
    15. }
    16. try {
    17. // 解析出扫描到类的元数据信息,里面包含了注解信息
    18. MetadataReader metadataReader = getMetadataReaderFactory().getMetadataReader(resource);
    19. // 关键方法,判断是否候选组件
    20. if (isCandidateComponent(metadataReader)) {
    21. ScannedGenericBeanDefinition sbd = new ScannedGenericBeanDefinition(metadataReader);
    22. sbd.setSource(resource);
    23. if (isCandidateComponent(sbd)) {
    24. if (debugEnabled) {
    25. logger.debug("Identified candidate component class: " + resource);
    26. }
    27. candidates.add(sbd);
    28. }
    29. else {
    30. if (debugEnabled) {
    31. logger.debug("Ignored because not a concrete top-level class: " + resource);
    32. }
    33. }
    34. }
    35. else {
    36. if (traceEnabled) {
    37. logger.trace("Ignored because not matching any filter: " + resource);
    38. }
    39. }
    40. }
    41. catch (FileNotFoundException ex) {
    42. if (traceEnabled) {
    43. logger.trace("Ignored non-readable " + resource + ": " + ex.getMessage());
    44. }
    45. }
    46. catch (Throwable ex) {
    47. throw new BeanDefinitionStoreException(
    48. "Failed to read candidate component class: " + resource, ex);
    49. }
    50. }
    51. }
    52. catch (IOException ex) {
    53. throw new BeanDefinitionStoreException("I/O failure during classpath scanning", ex);
    54. }
    55. return candidates;
    56. }
    57. 复制代码
    1. // 判断是否候选的Bean Component
    2. protected boolean isCandidateComponent(MetadataReader metadataReader) throws IOException {
    3. // exclude过滤器,在exclude过滤其中的,会直接排除掉,返回false
    4. for (TypeFilter tf : this.excludeFilters) {
    5. if (tf.match(metadataReader, getMetadataReaderFactory())) {
    6. return false;
    7. }
    8. }
    9. // include过滤器, 这里会看到有AnnotationTypeFilter,注解类型过滤器
    10. for (TypeFilter tf : this.includeFilters) {
    11. // 调用AnnotationTypeFilter的match方法,来判断是否满足条件
    12. if (tf.match(metadataReader, getMetadataReaderFactory())) {
    13. // 下面在进行Condition的判断,就是类上的@Conditional,这里不是重点
    14. return isConditionMatch(metadataReader);
    15. }
    16. }
    17. return false;
    18. }
    19. 复制代码

    而这个AnnotationTypeFilter默认是在构造函数中注册进去的。

    小结:

    @Component到Spring bean容器管理过程如下:

    1. 初始化时设置了Component类型过滤器;
    2. 根据指定扫描包扫描.class文件,生成Resource对象;
    3. 解析.class文件并注解归类,生成MetadataReader对象;
    4. 使用第一步的注解过滤器过滤出有@Component类;
    5. 生成BeanDefinition对象;
    6. 把BeanDefinition注册到Spring容器。

    总结

    经过这篇文章文章,是不是对@Component的使用和实现原理一清二楚了呢,其实Spring中还有@Service、@Controller和@Repository等注解,他们和@Component有什么区别呢?


     

  • 相关阅读:
    cat命令应用
    剑指 Offer 29. 顺时针打印矩阵
    Qt实战案例(59)——利用QTimer类实现定时器功能
    [详细]Apache Shiro身份验证越权漏洞[CVE-2016-6802][CVE-2020-1957]
    新的按人口比例的邮政编码
    unordered_map和unordered_set模拟实现
    (附源码)ssm学生选课系统 毕业设计 170920
    14:00面试,14:06就出来了,问的问题有点变态。。。
    算法训练第五十九天
    CSS中的 5 类常见伪元素详解!
  • 原文地址:https://blog.csdn.net/m0_71777195/article/details/126365263