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全面解析!

精彩视频

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Mysql 无法启动,mysql-bin.日志丢失删除处理

在linux操作系统中&#xff0c;当mysql无法启动时候&#xff0c;先看日志 2024-03-15T05:20:16.352075Z 0 [Warning] [MY-000081] [Server] option max_allowed_packet: unsigned value 107374182400 adjusted to 1073741824. 2024-03-15T05:20:16.352156Z 0 [Warning] [MY-010…

Marshmallow,一个有点甜的Python库

前言 在许多场景中&#xff0c;我们常常需要执行Python对象的序列化、反序列化操作。例如&#xff0c;在开发REST API时&#xff0c;或者在进行一些面向对象化的数据加载和保存时&#xff0c;这一功能经常派上用场。 经常cv Python代码的臭宝&#xff0c;接触最多的应该是通过…

被军训到的两天

1.gradle7.6.1 1.安装gradle7.6.1,一定要注意的是&#xff0c;使用的JDK是否能用&#xff0c;比如gradle7.6.1用的是JDK11。 2. F:/sofer....是Gradle自己的仓库地址&#xff0c;注意不能和maven使用一样的仓库。 使用specified location,可以避免下本项目的gradle版本&…

ego - 人工智能原生 3D 模拟引擎——基于AI的3D引擎,可以做游戏、空间计算、元宇宙等项目

1. 产品概述:Ego是一款AI本地化的3D模拟引擎,旨在让非技术创作者通过自然语言生成逼真的角色、3D世界和交互式脚本。该平台提供了创建和分享游戏、虚拟世界和交互体验的功能。 2. 定位:Ego定位于解决开放世界游戏和模拟的三大难题:难以编写游戏脚本、非玩家角色无法展现人…

linux系统关闭防火墙和SELINUX及配置网络

一&#xff0c;关闭防火墙和SELINUX 当我们进入界面后&#xff0c;输入用户名root&#xff0c;以及密码&#xff0c;密码我们是看不见的 然后输入指令cat -n /etc/sysconfig/selinux &#xff08;注意空格&#xff09; 输入指令 vi /etc/sysconfig/selinux &#xf…

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

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

MongoDB实战面试指南:常见问题一网打尽

码到三十五 &#xff1a; 个人主页 心中有诗画&#xff0c;指尖舞代码&#xff0c;目光览世界&#xff0c;步履越千山&#xff0c;人间尽值得 ! MongoDB是一款流行的非关系型数据库&#xff0c;以其高效、可扩展的特性受到开发者的青睐。了解MongoDB的架构、存储引擎和数据结…

ETH共识升级之路

简介 根据我们之前的介绍&#xff0c;了解到ETH网络的共识方式&#xff0c;已经从 PoW 切换到了 PoS&#xff0c;今天我们就回顾下升级之路&#xff0c;以及升级带来的影响 最早的共识机制 PoW 以太坊创建之初采用了类似比特币的工作量证明机制&#xff0c;即矿工通过计算哈希函…

CircuitBreaker熔断器

CircuitBreaker熔断器 1、Hystrix目前也进入维护模式 ​ Hystrix是一个用于处理分布式系统的延迟和容错的开源库&#xff0c;在分布式系统里&#xff0c;许多依赖不可避免的会调用失败&#xff0c;比如超时、异常等&#xff0c;Hystrix能够保证在一个依赖出问题的情况下&…

C++演变历史

C 从 C 98 版本到今年确定的 C23&#xff0c;共经历了 6 个版本的迭代。上面这张图里&#xff0c;列出了每次版本更新变化的内容。顶部黑体字的大小说明了版本变化的大小&#xff0c;其中 C14 是字体最小&#xff0c;这个版本是 C11 的增量版本&#xff0c;之所以没有大的变动&…

Server-Sent Events (SSE) 实现从服务器到客户端的实时数据流

前期回顾 避免阻塞主线程 —— Web Worker 示例项目-CSDN博客https://blog.csdn.net/m0_57904695/article/details/136721297?spm1001.2014.3001.5501 目录 CSDN 彩色之外 &#x1f4dd; 前言 &#x1f6e0;️ 安装 ✂️ 运行服务器 ✂️ 运行index.html ♻️ 贡献…

引领短剧风尚,打造全新观影体验——短剧APP开发之旅

随着移动互联网的迅猛发展&#xff0c;短视频和短剧成为了大众休闲娱乐的新宠。为了满足用户对于高质量、快节奏内容的需求&#xff0c;我们决定开发一款全新的短剧APP&#xff0c;为用户带来前所未有的观影体验。 这款短剧APP将集合丰富多样的短剧资源&#xff0c;涵盖各种题…

了解什么是 Redis 的雪崩、穿透和击穿?Redis 崩溃之后会怎么样?系统该如何应对这种情况?如何处理 Redis 的穿透?

目录 一、面试官心理分析 二、面试题剖析 1. 缓存雪崩 2. 缓存穿透 3. 缓存击穿 一、面试官心理分析 其实这是问到缓存必问的&#xff0c;因为缓存雪崩和穿透&#xff0c;是缓存最大的两个问题&#xff0c;要么不出现&#xff0c;一旦出现就是致命性的问题&#x…

B/S基于云计算的云HIS智慧医院管理系统源码带电子病历编辑器

目录 一、系统概述 二、开发环境 三、系统功能 1、门诊部分 2、住院部分 3、电子病历 4、药物管理 5、统计报表 6、综合维护 7、运营运维 云HIS系统&#xff1a;病案首页 云his系统源码 SaaS应用 功能易扩 统一对外接口管理 现如今&#xff0c;大数据、云计算、移动…

【WPF】Canvas的Path画线 颜色值偏差和面积不准确

图像中左上角2个红色线段 颜色值有偏差&#xff0c;且线段高度不准确&#xff08;我设置的是Red,1&#xff09;。 解决方案&#xff1a; 1、PathGeometry PathFigure LineSegment 2、Stretch设置Fill <Path Stretch"Fill" Stroke"Red"…

Axure软件安装汉化教程

Axure软件安装汉化教程 一、准备教程 下载Axure的软件&#xff0c;并解压打开 二、安装过程 双击Axure软件的运行程序&#xff0c;修改安装程序的路径&#xff0c;默认下一步即可。 三、软件汉化 打开Axure的软件安装路径&#xff0c;将汉化包复制粘贴进入到Axure RP 9安装…

C语言从入门到实战————数组和指针的深入理解

前言 在C语言中&#xff0c;数组和指针有的密切得联系&#xff0c;因为数组名本身就相当于一个指针常量。指针是一个变量&#xff0c;专门用来存储另一个变量的内存地址&#xff0c;通过这个地址可以访问和操作该变量的值&#xff0c;同时也包括数组。数组是一组连续存储的同类…

【CKA模拟题】查询消耗CPU最多的Pod

题干 For this question, please set this context (In exam, diff cluster name) 对于此问题&#xff0c;请设置此上下文&#xff08;在考试中&#xff0c;diff 集群名称&#xff09; kubectl config use-context kubernetes-adminkubernetesFind the pod that consumes the …

QT信号与槽实现方式

1、第一种实现方式 在QT开发工具UI界面先拖入按钮&#xff0c;然后鼠标右键拖入按钮&#xff0c;点击选中槽&#xff0c;在页面选着需要的信号&#xff0c;然后OK&#xff0c;随即将会跳转到类的.cpp文件&#xff0c;&#xff08;这种UI代码结合的方式&#xff0c;会自动去绑定…

在SwiftUI中使用Buider模式创建复杂组件

在SwiftUI中使用Buider模式创建复杂组件 我们在前面的博客闲聊SwiftUI中的自定义组件中聊到了如何在SwiftU中创建自定义组件。 在那里&#xff0c;我们创建了一个非常简单的组件RedBox&#xff0c;它将展示内容增加一个红色的边框。 RedBox非常简单&#xff0c;我们用普通的方…