Spring揭秘:ClassPathScanningProvider接口应用场景及实现原理!

Spring揭秘:ClassPathScanningCandidateComponentProvider接口应用场景及实现原理! - 程序员古德

技术应用场景

ClassPathScanningCandidateComponentProvider是Spring框架中一个非常核心的类,它主要用于在类路径下扫描并发现带有特定注解的组件,支持诸如@ComponentScan@Component@Service@Repository@Controller等注解的自动扫描和注册。

ClassPathScanningCandidateComponentProvider 解决了以下几个技术问题:

  1. 组件自动发现:在Spring应用程序中,会有大量的组件(如服务、控制器、存储库等),这些组件通常使用Spring的注解进行标记,手动配置这些组件可能会非常繁琐且容易出错,使用ClassPathScanningCandidateComponentProvider,Spring可以自动扫描类路径,发现并注册这些组件,从而大大简化了配置过程。
  2. 可扩展性:这个类提供了高度的可扩展性,可以通过覆盖其方法或提供自定义的过滤器来定制扫描过程,例如,可以指定只扫描特定包下的组件,或者只扫描带有特定注解的组件。
  3. 与Spring容器集成ClassPathScanningCandidateComponentProvider与Spring的ApplicationContext容器紧密集成,使用它发现的组件可以直接注册到容器中,使得这些组件能够在应用程序的其他部分中被自动装配和使用。
  4. 支持多种注解类型:这个类不仅支持Spring自带的注解(如@Component@Service等),还支持自定义注解,因此可以创建自己的注解,并使用ClassPathScanningCandidateComponentProvider在类路径中扫描带有这些注解的组件。

伪代码案例

下面是一个简单的示例,演示了如何使用 ClassPathScanningCandidateComponentProvider 类来扫描指定包路径下带有特定注解的类。

在这个例子中,使用带有 @Component 注解的类进行测试,如下代码。

首先,创建一些带有 @Component 注解的组件类作为扫描的目标。

// MyComponent1.java  
package com.example.components;  
  
import org.springframework.stereotype.Component;  
  
@Component  
public class MyComponent1 {  
    // ...  
}  
  
// MyComponent2.java  
package com.example.components;  
  
import org.springframework.stereotype.Component;  
  
@Component  
public class MyComponent2 {  
    // ...  
}

然后,编写一个客户端类,该类使用 ClassPathScanningCandidateComponentProvider 来扫描这些组件。

// ComponentScannerClient.java  
package com.example;  
  
import org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider;  
import org.springframework.core.type.filter.AnnotationTypeFilter;  
import org.springframework.stereotype.Component;  
  
import java.io.IOException;  
import java.util.Set;  
  
public class ComponentScannerClient {  
  
    public static void main(String[] args) {  
        // 创建一个 ClassPathScanningCandidateComponentProvider 实例  
        ClassPathScanningCandidateComponentProvider scanner = new ClassPathScanningCandidateComponentProvider(false);  
  
        // 添加一个过滤器,只包含带有 @Component 注解的类  
        scanner.addIncludeFilter(new AnnotationTypeFilter(Component.class));  
  
        // 指定要扫描的包路径  
        String basePackage = "com.example.components";  
  
        // 执行扫描并获取候选组件  
        Set<BeanDefinition> candidateComponents = scanner.findCandidateComponents(basePackage);  
  
        // 输出扫描结果  
        for (BeanDefinition beanDefinition : candidateComponents) {  
            System.out.println("Found component: " + beanDefinition.getBeanClassName());  
        }  
    }  
}

运行这个示例,控制台会输出类似下面的内容:

Found component: com.example.components.MyComponent1  
Found component: com.example.components.MyComponent2

核心API

ClassPathScanningCandidateComponentProvider 类提供了一系列的方法,用于配置扫描过程、执行扫描以及处理扫描结果。以下是该类中一些主要方法的含义:

  1. ClassPathScanningCandidateComponentProvider(boolean useDefaultFilters)
    • 构造函数,用于创建一个新的 ClassPathScanningCandidateComponentProvider 实例。
    • 参数 useDefaultFilters 指定是否应用默认的过滤器。如果为 true,则会自动包含对 @Component@Repository@Service@Controller 的支持。
  2. addIncludeFilter(TypeFilter includeFilter)
    • 添加一个包含过滤器,用于指定哪些类型的组件应该被包含在扫描结果中。
    • TypeFilter 是一个接口,可以通过实现该接口来定义自定义的过滤逻辑。
  3. addExcludeFilter(TypeFilter excludeFilter)
    • 添加一个排除过滤器,用于指定哪些类型的组件应该被排除在扫描结果之外。
    • 同样,TypeFilter 可以用于定义自定义的排除逻辑。
  4. findCandidateComponents(String basePackage)
    • 执行扫描操作,查找指定基础包及其子包下的候选组件。
    • 返回的是一个 Set<BeanDefinition>
  5. isCandidateComponent(MetadataReader metadataReader)
    • 判断给定的 MetadataReader 是否是一个候选组件。
    • 这个方法通常用于内部逻辑,但也可以被覆盖以实现自定义的候选组件判断逻辑。
  6. resetFilters(boolean includeDefaultFilters)
    • 重置之前添加的所有过滤器,并可以选择是否包含默认过滤器。
    • 这允许重用同一个 ClassPathScanningCandidateComponentProvider 实例进行多次不同的扫描操作。
  7. setEnvironment(Environment environment)
    • 设置用于解析属性占位符的 Environment 实例。
    • 这允许在扫描过程中使用 Spring 的环境抽象来解析例如占位符配置的值。
  8. setResourceLoader(ResourceLoader resourceLoader)
    • 设置用于加载资源的 ResourceLoader 实例。
    • 这允许在扫描过程中访问和加载类路径资源。
  9. setMetadataReaderFactory(MetadataReaderFactory metadataReaderFactory)
    • 设置用于创建 MetadataReader 实例的工厂。
    • 这允许自定义如何读取类的元数据。
  10. registerDefaultFilters()
    • 注册默认的过滤器。这个方法通常在构造函数中被调用,但也可以被覆盖以实现自定义的默认过滤器注册逻辑。

注意:这里列出的是一些核心方法,可能在不同的Spring版本中方法的数量会不一样,但是总体上差距不会非常大。

技术原理

ClassPathScanningCandidateComponentProvider类是Spring框架中用于扫描类路径以查找带特定注解的组件的关键类。它的实现原理基于Java的反射机制、类加载器以及Spring的内部元数据处理机制。

实现原理

1、类加载器和资源访问

  1. ClassPathScanningCandidateComponentProvider使用Java的类加载器(ClassLoader)来访问类路径上的资源,类加载器负责从文件系统、JAR文件或其他资源位置加载类。
  2. Spring利用类加载器的getResources方法来获取所有匹配给定包名的资源路径。

2、扫描和过滤

  1. 一旦获取了资源路径,ClassPathScanningCandidateComponentProvider就会扫描这些路径以查找候选组件。

  2. 扫描过程中,可以使用包含(include)和排除(exclude)过滤器来进一步细化扫描结果,这些过滤器基于注解、类名、包名或其他条件来过滤组件。

  3. 默认的过滤器通常会包含标注有@Component@Repository@Service@Controller注解的类。

3、元数据处理

  1. 对于扫描到的每个类,ClassPathScanningCandidateComponentProvider会使用MetadataReader来读取类的元数据,MetadataReader是Spring内部的一个接口,用于访问类的元数据而无需实际加载类。

  2. MetadataReaderFactory负责创建MetadataReader实例,默认情况下,它使用ASM库来读取类的字节码并提取元数据。

4、候选组件识别

  1. 通过分析元数据,ClassPathScanningCandidateComponentProvider能够确定一个类是否是一个候选组件,这是通过检查类上的注解来实现的。

  2. 如果一个类被识别为候选组件,它将被添加到扫描结果的集合中。

运行机制

底层算法大致如下:

  1. 初始化:配置 ClassPathScanningCandidateComponentProvider 实例,包括是否使用默认过滤器以及添加自定义的包含或排除过滤器。
  2. 扫描:调用 findCandidateComponents 方法并传入要扫描的基础包名,该方法内部会:1、使用类加载器的 getResources 方法获取所有匹配包名的资源路径。2、遍历这些资源路径,对每个路径执行扫描操作。
  3. 过滤和读取元数据:对于扫描到的每个资源(通常是 .class 文件),使用 MetadataReader 读取其元数据,并应用过滤器来确定是否应该将其包含为候选组件。
  4. 收集结果:将识别为候选组件的类信息收集起来并返回,这些信息通常用于后续的 Spring 容器初始化过程,如创建 Bean 定义等。

相关接口和类

  • TypeFilter:用于定义包含或排除过滤器的接口,实现该接口可以自定义过滤逻辑。
  • MetadataReaderMetadataReaderFactory:用于读取类的元数据而不实际加载类,这是通过直接访问类的字节码来实现的,通常使用 ASM 库来完成。
  • ResourceLoaderEnvironment:这些接口和类在扫描过程中不是必需的,但在某些情况下可以用于解析属性占位符或加载资源,它们提供了与 Spring 环境更紧密的集成。

关注我,每天学习互联网编程技术 - 程序员古德

END!
END!
END!

往期回顾

精品文章

Spring揭秘:Environment接口应用场景及实现原理!

Spring揭秘:AnnotationMetadata接口应用场景及实现原理!

Spring揭秘:BeanDefinitionBuilder接口应用场景及实现原理!

Spring揭秘:@import注解应用场景及实现原理!

Java并发基础:原子类之AtomicMarkableReference全面解析!

精彩视频

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/458692.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

.NET开源快速、强大、免费的电子表格组件

今天大姚给大家分享一个.NET开源&#xff08;MIT License&#xff09;、快速、强大、免费的电子表格组件&#xff0c;支持数据格式、冻结、大纲、公式计算、图表、脚本执行等。兼容 Excel 2007 (.xlsx) 格式&#xff0c;支持WinForm、WPF和Android平台&#xff1a;ReoGrid。 项…

普发Pfeiffer TPG256A MaxiGauge 真空计控制器接口通讯针脚等详情见图目录

普发Pfeiffer TPG256A MaxiGauge 真空计控制器接口通讯针脚等详情见图目录

强化学习中SARSA(State-Action-Reward-State-Action)和Q-learning的区别

SARSA&#xff08;State-Action-Reward-State-Action&#xff09;和Q-learning是两种经典的强化学习算法&#xff0c;它们都用于学习最优策略以使智能体在一个环境中获得最大的累积奖励。它们之间的主要区别在于它们更新动作值函数&#xff08;Q值函数&#xff09;的方式以及其…

SwiftUI组件-DatePicker

SwiftUI组件-DatePicker 本文记录一下SwiftUI组件-DatePicker import SwiftUIstruct DatePickerBootCamp: View {State var selectedDate: Date Date()var dateFormatter: DateFormatter {let formatter DateFormatter()formatter.dateStyle .shortformatter.timeStyle .…

使用kill()函数向进程发送信号

本片文章的学习记录总结来源于&#xff1a;https://www.bilibili.com/cheese/play/ep182660?csourcecommon_hp_history_null&t11&spm_id_from333.1007.top_right_bar_window_history.content.click 通常在Linux系统中&#xff0c;可以使用 kill or killall 命令向指定…

OpenCASCADE开发指南<十二>:OCC创建三维瓶子模型

在OpenCASCADE有一个例程&#xff0c;在 官方帮助网站中可以找到。程将教你如何使OpenCASCADE的API来进行三维建模。教程的目的不是描述所有的类&#xff0c;而是帮助你思考如何将OpenCASCADE作为一种工具。 1 概述 利用OpenCASCADE的API创建一个三维瓶子&#xff0c;形状如下…

如何在Linux部署DataEase数据分析服务并实现无公网IP远程分析内网数据信息

文章目录 前言1. 安装DataEase2. 本地访问测试3. 安装 cpolar内网穿透软件4. 配置DataEase公网访问地址5. 公网远程访问Data Ease6. 固定Data Ease公网地址 前言 DataEase 是开源的数据可视化分析工具&#xff0c;帮助用户快速分析数据并洞察业务趋势&#xff0c;从而实现业务…

IBM:《2024年消费者调研:无处不在的人工智能彻底变革零售业》

1月17日&#xff0c;IBM商业价值研究院最近发布了第三份两年一度的消费者调研报告。 这项名为《无处不在的人工智能彻底改变零售业&#xff1a;客户不会等待》的报告&#xff0c;对包含中国在内的全球近20000名消费者进行了调研&#xff0c;相关结果反映了消费者对零售体验的普…

【Python】进阶学习:一文了解NotImplementedError的作用

【Python】进阶学习&#xff1a;一文了解NotImplementedError的作用 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望…

2020-11-08 字符串指定位置倒序输出

缘由https://bbs.csdn.net/topics/398165677 //字符串指定位置倒序输出缘由https://bbs.csdn.net/topics/398165677char aa[] "abcABCabc\n";int a 3, b 5, c 0, d b;while (aa[c] ! \n)if (c < a || c > b)cout << aa[c]; else if(d > a) cout …

目标检测——YOLOv4算法解读

论文&#xff1a;YOLOv4&#xff1a;Optimal Speed and Accuracy of Object Detection 作者&#xff1a;Alexey Bochkovskiy, Chien-Yao Wang, Hong-Yuan Mark Liao 链接&#xff1a;https://arxiv.org/pdf/2004.10934.pdf 代码&#xff1a;https://github.com/AlexeyAB/darkne…

弹性盒子布局 Flexbox Layout

可以嵌套下去 1.display 属性 默认行排列 <style>.flex-item{ height: 20px;width: 10px;background-color: #f1f1f1;margin: 10px;}</style> </head> <body> <div class"flex-container"><div class"flex-item">1&l…

如何实现固定公网地址远程SSH连接Linux Deepin系统

文章目录 前言1. 开启SSH服务2. Deppin安装Cpolar3. 配置ssh公网地址4. 公网远程SSH连接5. 固定连接SSH公网地址6. SSH固定地址连接测试 前言 Deepin操作系统是一个基于Debian的Linux操作系统&#xff0c;专注于使用者对日常办公、学习、生活和娱乐的操作体验的极致&#xff0…

探索数据结构:双向链表的灵活优势

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ &#x1f388;&#x1f388;养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; 所属专栏&#xff1a;数据结构与算法 贝蒂的主页&#xff1a;Betty’s blog 1. 前言 前面我们学习了单链表&#xff0c;它解决了顺序表中插入删除需…

第110讲:Mycat实践指南:指定Hash算法分片下的水平分表详解

文章目录 1.应用指定Hash算法分片的概念2.使用应用指定Hash算法分片对某张表进行水平拆分2.1.在所有的分片节点中创建表结构2.2.配置Mycat实现应用指定Hash算法分片的水平分表2.2.1.配置Schema配置文件2.2.2.配置Rule分片规则配置文件2.2.3.配置Server配置文件2.2.4.重启Mycat …

什么牌子的蓝牙耳机质量好?2024年精选机型,真实体验分享

​对于新手来说&#xff0c;真无线蓝牙耳机的选购可能显得有些复杂。网络上有许多关于蓝牙耳机品牌、音质、舒适度的讨论。我整理了五款佩戴舒适且音质表现不错的真无线蓝牙耳机&#xff0c;希望能为你提供有价值的参考&#xff0c;不要错过哦&#xff01; 一、蓝牙耳机选购技巧…

训练YOLOv8m时AMP显示v8n

在训练Yolov8模型时&#xff0c;使用AMP&#xff08;Automatic Mixed Precision&#xff09;可以加速训练过程并减少显存的使用。AMP是一种混合精度训练技术&#xff0c;它通过将模型参数的计算转换为低精度&#xff08;如半精度&#xff09;来提高训练速度&#xff0c;同时保持…

es 分词器详解

基本概念 分词器官方称之为文本分析器&#xff0c;顾名思义&#xff0c;是对文本进行分析处理的一种手段&#xff0c;基本处理逻辑为按照预先制定的分词规则&#xff0c;把原始文档分割成若干更小粒度的词项&#xff0c;粒度大小取决于分词器规则。 分词器发生的时期 1、分词…

pytorch之诗词生成6--eval

先上代码&#xff1a; import tensorflow as tf from dataset import tokenizer import settings import utils# 加载训练好的模型 model tf.keras.models.load_model(r"E:\best_model.h5") # 随机生成一首诗 print(utils.generate_random_poetry(tokenizer, model)…

微信公众号测试号里面显示若依前端页面

内网穿透 注册购买内网穿透隧道 https://natapp.cn/ 启动成功 这样就绑定你的本地启动项目 微信公众测试号配置 注册微信公众号测试号 获取access_token&#xff0c;AppID与appsecret 调用微信官方接口生成access_token&#xff08;AppID和AppSecret可在“微信公众平台-设置…