Spring源码注解篇二:手写@Component注解

@Component注解的功能

在Spring框架中,@Component 注解是一个核心特性,用于自动检测类并将其注册为Spring应用上下文中的Bean。这大大简化了Bean的配置过程,使得开发者能够通过注解的方式快速地将类标记为组件,并由Spring容器进行管理。本文将首先探讨@Component注解的定义,然后设计并实现一个自定义的@MyselfComponent注解,该注解将具备与@Component相同的功能,即自动创建并管理对应的实例。

一、Spring中@Component注解的定义

在Spring的源码中,@Component注解本身并没有直接实现Bean的创建和管理逻辑,它主要是一个标记注解,用于被Spring的组件扫描机制识别。@Component注解通常与@ComponentScan注解一起使用,后者指示Spring框架在哪些包中查找带有@Component(或它的衍生注解如@Service@Repository@Controller)的类,并将这些类实例化为Bean。

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Indexed
public @interface Component {String value() default "";
}

从这个定义可以看出,@Component注解非常简单,主要标记了注解的目标类型(TYPE,即类、接口或枚举类型),并定义了注解的保留策略(RUNTIME),以便在运行时通过反射访问注解信息。

二、设计自定义@MyselfComponent注解

为了设计一个与@Component功能相似的自定义注解@MyselfComponent,我们需要考虑以下几点:

  1. 注解定义:定义@MyselfComponent注解,使其能够标记类。
  2. 组件扫描:实现或集成一个组件扫描机制,用于识别并处理带有@MyselfComponent注解的类。
  3. Bean注册:将扫描到的类实例化为Bean,并注册到Spring应用上下文中。

注解定义

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface MyselfComponent {String value() default "";
}

组件扫描与Bean注册

由于Spring框架提供了丰富的API来处理组件扫描和Bean注册,我们可以利用这些API来简化我们的实现。但为了演示目的,我们将手动模拟这一过程。

三、动手实现@MyselfComponent注解的功能

在实际项目中,我们通常不会完全从头开始实现Spring的组件扫描和Bean注册逻辑,因为这需要深入了解Spring的底层架构。但在这里,我们将通过创建一个简单的模拟框架来展示这一过程。

步骤 1: 创建模拟的ApplicationContext

import java.util.HashMap;
import java.util.Map;public class SimpleApplicationContext {private Map<String, Object> beans = new HashMap<>();public void registerBean(String name, Object bean) {beans.put(name, bean);}public <T> T getBean(String name, Class<T> requiredType) {return (T) beans.get(name);}// 模拟扫描并注册Bean的逻辑...
}

步骤 2: 实现扫描逻辑

这里我们仅模拟扫描过程,不深入Java反射和类路径扫描的具体实现。

public class SimpleComponentScanner {private SimpleApplicationContext context;public SimpleComponentScanner(SimpleApplicationContext context) {this.context = context;}public void scan(String packageName) {// 这里应该使用反射和类加载器来查找指定包下的类,并检查是否带有@MyselfComponent注解// 但为了简化,我们直接模拟注册context.registerBean("myService", new MyService()); // 假设MyService类上有@MyselfComponent注解}
}

步骤 3: 使用

@MyselfComponent
public class MyService {// 类的实现...
}public class Main {public static void main(String[] args) {SimpleApplicationContext context = new SimpleApplicationContext();SimpleComponentScanner scanner = new SimpleComponentScanner(context);scanner.scan("org.springframework.annotation"); // 假设MyService位于这个包下MyService myService = context.getBean("myService", MyService.class);// 使用myService...}
}

结论

虽然上述实现非常基础且远未达到Spring框架的复杂性和功能全面性,但它为理解@Component注解的工作原理以及如何在自定义框架中模拟类似功能提供了一个基本的框架。在实际应用中,Spring的组件扫描和Bean注册过程要复杂得多,它涉及到多个组件的协同工作,包括ClassPathScanningCandidateComponentProviderBeanDefinitionRegistryAnnotationBeanNameGenerator等。

深化理解

为了更深入地理解并模拟Spring的组件扫描和Bean注册过程,我们可以考虑以下几个关键点:

  1. 使用Java反射API

    • 遍历指定包下的所有类。
    • 检查每个类上是否存在@MyselfComponent注解。
    • 如果存在,则创建该类的实例,并生成一个合适的Bean名称(通常基于类名)。
  2. 注册Bean到应用上下文

    • 在Spring中,这通常涉及到操作BeanDefinitionRegistry,但在我们的模拟框架中,我们可能只是简单地将Bean实例存储在一个Map中。
    • 需要确保Bean的注册过程支持依赖注入、生命周期管理等高级功能(尽管在我们的简化示例中未实现这些)。
  3. 处理注解的属性

    • @MyselfComponent注解中的value属性可以用来指定Bean的名称。如果没有指定,可以生成一个默认名称(如类名的首字母小写形式)。
  4. 集成Spring的现有机制(可选):

    • 如果可能,可以尝试集成Spring的ClassPathScanningCandidateComponentProvider等类来简化扫描过程。
    • 使用Spring的AnnotationConfigApplicationContextAnnotationBeanNameGenerator等类来更紧密地模拟Spring的行为。

示例扩展

以下是一个更接近于实际Spring行为的SimpleComponentScanner实现示例,尽管它仍然使用了简化的Bean注册方式:

import org.springframework.core.type.filter.AnnotationTypeFilter;
import org.springframework.util.ClassUtils;import java.lang.reflect.Modifier;
import java.util.Set;// 假设我们使用Spring的ClassPathScanningCandidateComponentProvider来扫描类
// 但为了简化,这里只描述如何调用它,并不实现完整的类
public class AdvancedComponentScanner {private SimpleApplicationContext context;public AdvancedComponentScanner(SimpleApplicationContext context) {this.context = context;}public void scan(String basePackage) {// 这里应该使用ClassPathScanningCandidateComponentProvider来扫描// 但为了简化,我们直接模拟扫描到的类Class<?> clazz = MyService.class; // 假设MyService是我们扫描到的类if (clazz.isAnnotationPresent(MyselfComponent.class) && !Modifier.isAbstract(clazz.getModifiers())) {String beanName = clazz.getSimpleName().substring(0, 1).toLowerCase() + clazz.getSimpleName().substring(1);context.registerBean(beanName, clazz.getDeclaredConstructor().newInstance());}// 注意:这里省略了异常处理、依赖注入等复杂逻辑}
}

注意:上述代码中的clazz.getDeclaredConstructor().newInstance()是不推荐的做法,因为它不处理任何可能的异常(如NoSuchMethodExceptionIllegalAccessExceptionInstantiationExceptionInvocationTargetException),并且从Java 9开始,newInstance()方法已被标记为过时。在实际应用中,应该使用Constructor.newInstance(Object... initargs)的替代方法,如通过Class.getDeclaredConstructor().newInstance(initargs)(需要处理异常)或使用更现代的反射API,如MethodHandles.Lookup

此外,上面的代码也没有处理类加载器的问题,这在处理复杂的应用程序和类路径时非常重要。在Spring中,这些都被优雅地处理了。

总之,模拟Spring的@Component注解功能是一个很好的学习练习,它可以帮助我们更深入地理解Spring的IoC容器和组件扫描机制。然而,要完全实现Spring的功能,需要深入了解Spring的源码和Java的反射API。

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

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

相关文章

硬件检测工具 | CPU-Z v2.10.0 官方中文绿色版

软件简介 CPU-Z是一款广受欢迎的硬件检测工具&#xff0c;主要用于收集电脑处理器的详细信息。这款软件能够提供关于CPU的详细数据&#xff0c;包括处理器名称、编号、代号、进程和缓存等信息。此外&#xff0c;CPU-Z还能实时监测每个内核的内部频率和内存频率&#xff0c;以及…

【洛谷B3644】【模板】拓扑排序 / 家谱树 解题报告

洛谷B3644 - 【模板】拓扑排序 / 家谱树 题目描述 有个人的家族很大&#xff0c;辈分关系很混乱&#xff0c;请你帮整理一下这种关系。给出每个人的后代的信息。输出一个序列&#xff0c;使得每个人的后辈都比那个人后列出。 输入格式 第 1 1 1 行一个整数 N N N&#xf…

Java字节流的输入输出

Java字节流的输入输出指的是使用Java中的字节流来进行数据的读取和写入操作。字节流是以字节为单位进行读写操作的&#xff0c;常用于处理二进制数据或者字符数据。 Java中常用的字节流类有InputStream和OutputStream。InputStream用于读取字节数据&#xff0c;OutputStream用…

【运维项目经历|037】MFS-Scale分布式对象存储系统部署与优化项目

🍁博主简介: 🏅云计算领域优质创作者 🏅2022年CSDN新星计划python赛道第一名 🏅2022年CSDN原力计划优质作者 ​ 🏅阿里云ACE认证高级工程师 ​ 🏅阿里云开发者社区专家博主 💊交流社区:CSDN云计算交流社区欢迎您的加入! 目录 项目名称 项目背景 项目目标 …

如何在项目中打印sql和执行的时间

目标&#xff1a;打印DAO方法中sql和执行的时间 一种方式是去实现Mybatis的拦截器Interceptor &#xff0c;比较麻烦&#xff1b; 这里介绍一种比较简单的实现方式&#xff1b; 1、如何打印sql&#xff1f; 配置文件加这个可以打印出com.zhenhui.ids.busi.watch包下执行的sq…

c++包管理器

conan conan search&#xff0c;查看网络库 conan profile detect&#xff0c;生成缓存信息conan new cmake_exe/cmake_lib&#xff0c;创建cmakelists.txtconan install .&#xff0c;执行Conanfile.txt中的配置&#xff0c;生成相关的bat文件 项目中配置Conanfile.txt(或者…

ImportError: DLL load failed while importing _swigfaiss: 找不到指定的模块。

ImportError: DLL load failed while importing _swigfaiss: 找不到指定的模块 这个错误通常是由于系统中缺少某些必要的动态链接库&#xff08;DLL&#xff09;文件&#xff0c;或者由于与当前环境中的库版本不兼容导致的。以下是一些解决这个问题的步骤&#xff1a; 方法一&…

linux高级编程(sqlite数据库调用)

数据库 1、分类&#xff1a; 大型 中型 小型 ORACLE MYSQL/MSSQL SQLITE DBII powdb 关系型数据库 2、名词&#xff1a; DB 数据库 select update database DBMS 数据库管理系统 MIS 管理…

tessy 集成测试:小白入门指导手册

目录 1,创建集成测试模块且分析源文件 2,设置测试环境 3,TIE界面设置相关函数 4,SCE界面增加用例 5,编辑数据 6,用例所对应的测试函数序列 7,添加 work task 函数 8,为测试场景添加函数 9,为函数赋值 10,编辑时间序列的数值 11,执行用例 12,其他注意事项…

《昇思25天学习打卡营第2天|初学教程/快速入门》

文章目录 快速入门处理数据集网络构建模型训练保存模型加载模型 快速入门 通过MindSpore的API来快速实现一个简单的深度学习模型。 MindSpore 是华为推出的一个全场景深度学习框架&#xff0c;它旨在实现易开发、高效执行和全场景覆盖的目标。这个框架支持云、边缘以及端侧场…

零基础STM32单片机编程入门(十四) DS18B20温度传感器模块实战含源码

文章目录 一.概要二.DS18B20主要性能参数三.DS18B20温度传感器内部框图四.DS18B20模块原理图五.DS18B20模块跟单片机板子接线和通讯时序1.单片机跟DS18B20连接示意图2.单片机跟DS18B20通讯流程与时序3.通讯流程中的9个数据字节内容格式4.温度数据寄存器LSB/MSB格式 六.STM32单片…

ROS2入门到精通—— 1-3 ROS1移植到ROS2系统梳理

ROS2同一功能包只能同时包含Python或者C一种 1 ROS1 && ROS2 CMakeList.txt ROS1中CMakeLists.txt架构如下&#xff1a; cmake_minimum_required() #CMake的最低版本号project() #项目名称find_package() #找到编译需要的其他CMake/Catkin package catkin_py…

【HarmonyOS】鸿蒙中如何获取用户相册图片?photoAccessHelper.PhotoViewPicker

【HarmonyOS】鸿蒙中如何获取用户相册图片&#xff1f;photoAccessHelper.PhotoViewPicker 前言 有同学私聊我说&#xff0c;之前的博客文章提到的没有HarmonyOS白名单帐号&#xff0c;如何在OpenHarmony Gitee开发仓里学习API接口。需要注意一个点&#xff0c;默认看到的文档…

图、图的遍历、最小生成树、最短路径

0、图的概念 **图:**是由顶点集合及顶点间的关系组成的一种数据结构&#xff1a;G (V&#xff0c; E)&#xff0c;其中&#xff1a; 顶点集合V {x|x属于某个数据对象集}是有穷非空集合&#xff1b;边的集合E {(x,y)|x,y属于V}或者E {|x,y属于V && Path(x, y)}是顶…

浅学三次握手

数据要完成传输&#xff0c;必须要建立连接。由于建立TCP连接的过程需要来回3次&#xff0c;所以&#xff0c;将这个过程形象的叫做三次握手。 结合上面的图来看更清楚。 先说三次握手吧&#xff0c;连接是后续数据传输的基础。就像我们打电话一样&#xff0c;必须保证我和对方…

在 PostgreSQL 里如何实现数据的实时监控和性能瓶颈的快速定位?

&#x1f345;关注博主&#x1f397;️ 带你畅游技术世界&#xff0c;不错过每一次成长机会&#xff01;&#x1f4da;领书&#xff1a;PostgreSQL 入门到精通.pdf 文章目录 在 PostgreSQL 里如何实现数据的实时监控和性能瓶颈的快速定位一、数据实时监控的重要性二、PostgreSQ…

shell,重定向与管道符号

文章目录 一&#xff0c;什么是shell二&#xff0c;shell脚本和作用1. shell脚本2. 作用 三&#xff0c;shell脚本的构成内容四&#xff0c;创建和运行Shell脚本五&#xff0c;重定向与管道操作1. 交互式硬件设备2. 重定向操作概览表3. 管道操作 一&#xff0c;什么是shell sh…

Matlab 判断直线上一点

文章目录 一、简介二、实现代码三、实现效果参考资料一、简介 判断一个点是否位于一直线上有很多方法,这里使用一种很有趣的坐标:Plucker线坐标,它的定义如下所示: 这个坐标有个很有趣的性质,我们可以使用Plucker坐标矢量构建一个Plucker矩阵: 则它与位于对应线上的齐次点…

排序(三)——归并排序(MergeSort)

欢迎来到繁星的CSDN&#xff0c;本期内容主要包括归并排序(MergeSort)的实现 一、归并排序的主要思路 归并排序和上一期讲的快速排序很像&#xff0c;都利用了分治的思想&#xff0c;将一整个数组拆成一个个小数组&#xff0c;排序完毕后进行再排序&#xff0c;直到整个数组排序…

文章管理小程序的设计

管理员账户功能包括&#xff1a;系统首页&#xff0c;个人中心&#xff0c;作者管理&#xff0c;文章管理&#xff0c;文章分类管理&#xff0c;论坛&#xff0c;系统管理 微信端账号功能包括&#xff1a;系统首页&#xff0c;文章&#xff0c;论坛&#xff0c;我的 开发系统…