手写简单实现IOC

news2025/1/23 6:07:53

这个小demo是利用反射从最基础一步一步模拟实现了IOC的功能,所有的代码基本都给出了注释,方便大家阅读.

目录结构:

这里需要导入一下junit依赖

   <!-- junit测试 -->
        <dependency>
            <groupId>junit</groupId>
            <artifactId>junit</artifactId>
            <version>4.12</version>
            <scope>test</scope>
        </dependency>

 两个工具类:

TestController类:

package com.heaboy.springioc.entity;

import com.heaboy.springioc.stereotype.Autowired;
import com.heaboy.springioc.stereotype.Component;
import sun.misc.Contended;

@Component
public class TestController {
    @Autowired
    private UserService userService;
    public void test(){
        //通过反射拿到userService对象调用addUser方法
        userService.addUser("zhangsan",111);
    }
}

UserService类:

package com.heaboy.springioc.entity;

import com.heaboy.springioc.stereotype.Component;

@Component
public class UserService {
    //接受传参 并且输出
    public void addUser(String name,int age){
        System.out.println(name);
        System.out.println(age);
    }
}

两个注解类:

Autowired:

package com.heaboy.springioc.stereotype;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.FIELD)
@Retention(RetentionPolicy.RUNTIME)
public @interface Autowired {
}

Component:

package com.heaboy.springioc.stereotype;

import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;

@Target(ElementType.TYPE)
@Retention(RetentionPolicy.RUNTIME)
public @interface Component {
}

最重要的模拟IOC类:

package com.heaboy.springioc.ioc;

import com.heaboy.springioc.stereotype.Autowired;
import com.heaboy.springioc.stereotype.Component;

import java.io.File;
import java.io.FileNotFoundException;
import java.lang.annotation.Annotation;
import java.lang.reflect.Field;
import java.util.*;

public class SpringIOC {
    //声明存放bean名称的列表
    private List<String> beanNames;
    //声明存放文件路径的列表
    private List<String> filePaths;
    //声明基础路径字符串
    private String basePath;
    //声明基础包名字符串
    private String basePackage;
    //声明存放bean对象的映射
    private Map<String,Object> beans = new HashMap<>();
    /**
     * 扫描所有的文件信息,存放到filePaths
     */
    private void scan() throws FileNotFoundException{
        //创建File 对象表示基础路径
        File file = new File(basePath);
        //初始化存放文件路径的列表
        filePaths = new ArrayList<>();
        //判断文件是否存在
        if(file.exists()){
            //定义一个列表 用于逐层深入的遍历每个文件(广度优先搜索)
            Queue<File> queue = new LinkedList<>();
            //将根文件添加到队列当中
            queue.add(file);
            //判断队列是否为空
            while (!queue.isEmpty()){
                //取出 根文件赋值给poll便于逐层深入
                File poll = queue.poll();
                //如果poll是空说明他下面已经不存在文件和文件路径了 直接跳过执行下一步
                if(poll == null){
                    continue;
                }
                //如果是文件夹
                if(poll.isDirectory()){
                    //创建一个File类型的数组 拿到poll里面的全部文件,文件夹
                    File[] files = poll.listFiles();
                    for (File file1:files){
                        //将poll里面的全部文件夹以及文件加入队列
                        queue.add(file1);
                    }
                    //如果此poll是文件
                }else {
                    //直接将文件的路径添加到filePaths列表中
                    filePaths.add(poll.getPath());
                }
            }
        }else {
            //如果没找到文件
            //抛出文件未找到异常
            throw new FileNotFoundException(basePath+"not found");
        }
    }
    /**
     * 将所有的.java结尾的 全限定名放到 beanNames 列表里面
     */
    public void initBeanNames(){
        //遍历 filePaths里面的每一个文件的路径
        for (String s :filePaths){
            //将文件中的 基础路径部分替换成空串(因为所有文件路径的基础路径都是一样的,
            // 我们需要获取的是以.java结尾的文件的全限定名 前缀需要去掉)
            String replace = s.replace(basePath,"");
            //判断文件的全限定名是不是以.java结尾
            if (replace.endsWith(".java")){
                //截取掉.java
                replace = replace.substring(0,replace.length()-5);
            }
            //将字符串转换成字符数组 目的是为了方便把路径中的\\替换成.
            char[] chars = replace.toCharArray();
            for (int i=0;i<chars.length;i++){
                if(chars[i]=='\\'){
                    chars[i]='.';
                }
            }
            //拼接字符串 拼接出文件在该项目中的源根地址
            beanNames.add(basePackage+"."+new String(chars));
        }
    }
    /**
     * 初始化所有的Bean对象 根据上个方法求出来的所有文件的源根地址
     * 遍历每一个文件(其实就是java文件) 看四否含有Component注解(也就是被标记为
     * 它表明这个类会被 Spring IoC 容器自动检测和注册为 Spring 的一个 bean。
     * 也就是把类交给spring管理)
     */
    public void initBeans(){
        for (String beanName : beanNames){
            try {
                //根据路径 拿到具体的类对象
                Class<?> aClass = Class.forName(beanName);
                //拿到这个类声明的所有注解
                Annotation[] declaredAnnotations = aClass.getDeclaredAnnotations();
                //遍历所有注解 看是否存在@Component注解
                for (Annotation declaredAnnotation : declaredAnnotations){
                    //如果是Component注解
                    if(declaredAnnotation instanceof Component){
                        //创建该类的实例化对象
                        Object o =aClass.newInstance();
                        //将对象放入beans映射中
                        //aClass.getName() 类名  o 实例化对象
                        beans.put(aClass.getName(),o);
                    }
                }
            } catch (ClassNotFoundException e) {
                e.printStackTrace();
            } catch (InstantiationException e) {
                e.printStackTrace();
            } catch (IllegalAccessException e) {
                e.printStackTrace();
            }
        }
        //依次遍历 查看是否有Autowired注解 如果有,给对应字段附上相应的实例化对象
        for (Map.Entry<String,Object> entry : beans.entrySet()){
            //利用反射获取到对象对应类的所有声明字段
            Field[] declaredFields = entry.getValue().getClass().getDeclaredFields();
            //遍历所有字段 获取每个字段声明的注解看是否包含Autowired注解
            for (Field field : declaredFields){
                Annotation[] annotations = field.getDeclaredAnnotations();
                for (Annotation annotation : annotations){
                    if(annotation instanceof Autowired){
                        //是Autowired注解我们需要为对应的变量 付给他对应的对象
                        //先获取该字段的类型对应的对象名称
                        String name = field.getType().getName();
                        //从beans里获取这个名字的对象
                        Object o = beans.get(name);
                        //设置字段可访问
                        field.setAccessible(true);
                        try {
                            //给字段赋值
                            //也就是说 原本没有赋值的时候entry对象里对应的字段的值是一个空的对象,
                            //然后通过set方法把这个字段的值替换成相应的对象
                            field.set(entry.getValue(),o);
                        } catch (IllegalAccessException e) {
                            e.printStackTrace();
                        }
                    }
                }
            }
        }


    }
    /**
     * 根据bean名称获取对应的实例化对象
     * beanName bean的名称
     * 返回值:一个实例化对象
     */
    public Object getInstance(String beanName){
        return beans.get(beanName);
    }

    /**
     * 初始化基础路径和包名
     */
    private void initPath(){
        //初始化文件的路径 你项目所在电脑中的位置
        basePath="D:\\blog1\\ioc\\src\\main\\java\\com\\heaboy\\springioc\\";
        //初始化基础包名
        basePackage = "com.heaboy.springioc";
    }
    /**
     * SpringIOC类的构造方法,初始化路径和扫描文件
     */
    public SpringIOC(){
        //初始化路径和包名
        initPath();
        try {
            scan();//扫描文件
        } catch (FileNotFoundException e) {
            e.printStackTrace();
        }
        //初始化bean名称列表
        beanNames = new ArrayList<>();
        //初始化bean名称
        initBeanNames();
    }

}

测试类:

package com.heaboy.springioc.ioc;


import com.heaboy.springioc.entity.TestController;
import org.junit.Test;


import java.io.FileNotFoundException;

public class SpringIOCTest {
    @Test
    public void testScan() throws FileNotFoundException {
        //创建SpringIOC类型的实例化对象
        SpringIOC springIOC = new SpringIOC();
        //调用SpringIOC的bean对象初始化方法
        springIOC.initBeans();
        //借用SpringIOC中的利用反射创建实例化对象方法 通过传入一个类名获取到实例化对象
        TestController instance = (TestController)springIOC.getInstance(TestController.class.getName());
        //对象调用test方法;
        instance.test();

    }
}

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

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

相关文章

string类(STL开始)

相信大家都知道STL在C中的重要性&#xff0c;作为其模板库中的一部分&#xff0c;包含了常见的数据结构和算法&#xff0c;是C的标准库 而我们今天要讲的String类&#xff08;String底层是一个字符顺序数组的顺序表对象&#xff0c;可以归类为容器&#xff09;&#xff0c;其实…

JavaWeb系列二十二: 线程数据共享和安全(ThreadLocal)

韩顺平-线程数据共享和安全ThreadLocal 什么是ThreadLocal?ThreadLocal环境搭建ThreadLocal快速入门ThreadLocal源码阅读threadLocal.set()源码threadLocal.get()源码 什么是ThreadLocal? ThreadLocal的作用: 可以实现在同一个线程数据共享, 从而解决多线程数据安全问题.Thr…

一.6 存储设备形成层次结构

在处理器和一个较大的较慢的设备&#xff08;例如主存&#xff09;之间插入一个更小更快的存储设备&#xff08;例如高速缓存&#xff09;的想法已经成为一个普遍的概念。实际上&#xff0c;每个计算机系统重的存储设备都被组织成了一个存储器层次结构&#xff0c;如图1-9所示。…

楼梯导航案例

楼梯导航 <!DOCTYPE html> <html lang"zh-cn"><head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>滚动到展示区</title><link re…

Python 中将字典内容保存到 Excel 文件使用详解

概要 在数据处理和分析的过程中,经常需要将字典等数据结构保存到Excel文件中,以便于数据的存储、共享和进一步分析。Python提供了丰富的库来实现这一功能,其中最常用的是pandas和openpyxl。本文将详细介绍如何使用这些库将字典内容保存到Excel文件中,并包含具体的示例代码…

NodeJS小饰品销售管理系统-计算机毕业设计源码21597

摘 要 在当今的数字化时代&#xff0c;电子商务已经成为了商业领域中不可或缺的一部分。随着消费者对于购物体验的要求越来越高&#xff0c;一个高效、便捷、用户友好的小饰品销售管理系统显得尤为重要。 本系统旨在利用 JavaScript 技术&#xff0c;设计并实现一个功能强大的小…

Python实现动态银河系:模拟旋转的银河动画

文章目录 引言准备工作前置条件 代码实现与解析导入必要的库初始化Pygame定义星系类主循环 完整代码 引言 银河系的旋转动画是一个迷人且富有挑战性的项目。通过模拟星系的旋转&#xff0c;我们可以更好地理解天文学现象&#xff0c;并创造出视觉上令人惊叹的效果。在这篇博客…

计算机网络 - 万字长文

计算机网络 二、计算机网络2.1 七层模型表格2.2 通俗讲解七层模型2.3 TCP与UDP对比2.4 TCP 三次握手过程==为什么握手是三次,而不是两次或者四次?====三次握手可以携带数据吗?====TCP三次握手失败,服务端会如何处理?====什么是半连接队列?全连接====ISN(Initial Sequence…

昇思MindSpore学习入门-CELL与参数一

Cell作为神经网络构造的基础单元&#xff0c;与神经网络层(Layer)的概念相对应&#xff0c;对Tensor计算操作的抽象封装&#xff0c;能够更准确清晰地对神经网络结构进行表示。除了基础的Tensor计算流程定义外&#xff0c;神经网络层还包含了参数管理、状态管理等功能。而参数(…

【Python】已解决:(最新版selenium框架元素定位报错)NameError: name ‘By’ is not defined

文章目录 一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项 已解决&#xff1a;&#xff08;最新版selenium框架元素定位报错&#xff09;NameError: name ‘By’ is not defined 一、分析问题背景 在使用Selenium进行Web自动化测试或爬虫开…

R包:‘patchwork合并多个R图的包‘

介绍 patchwork是基于gglot2的拼图包&#xff0c;它使得基于ggplot2的图形更容易拼接在同一个图层。 安装 因为作者仅仅在GitHub发布了patchwork&#xff0c;因此无法使用install.packages("patchwork")从CRAN处获取。为了获取该包&#xff0c;首先应该安装devtoo…

十进制与十六进制,和二进制的相互转变

十六进制与十进制 十六进制&#xff08;Hexadecimal&#xff09;是一种进位制&#xff0c;基数为16&#xff0c;常用于计算机科学和电子工程中。十六进制使用16个符号来表示数值&#xff1a;0-9表示0到9&#xff0c;A-F表示10到15。十六进制的每一位可以表示4位二进制数&#…

万界星空科技MES:磷酸铁锂正极新材料生产管理系统

磷酸铁锂MES通过对生产现场的数据进行实时采集、处理和监控&#xff0c;实现对生产过程的优化和控制。它可以实时监控生产设备的运行状态、物料的使用情况、产品的生产进度等信息&#xff0c;并根据这些信息对生产过程进行调整和优化。例如&#xff0c;当发现某个生产设备的故障…

电脑桌面日历记事本怎么弄 好用的桌面日历记事本

在这个数字化的时代&#xff0c;电脑已成为我们日常生活中不可或缺的伙伴。我常常在电脑上记录各种事项&#xff0c;以便随时查看和提醒自己。而我最钟爱的记事方式&#xff0c;莫过于使用桌面日历记事本。 想象一下&#xff0c;你的电脑桌面上有一个直观的日历&#xff0c;每…

【Elasticsearch】开源搜索技术的演进与选择:Elasticsearch 与 OpenSearch

开源搜索技术的演进与选择&#xff1a;Elasticsearch 与 OpenSearch 1.历史发展2.OpenSearch 与 Elasticsearch 相同点3.OpenSearch 与 Elasticsearch 不同点3.1 版本大不同3.2 许可证不同3.3 社区不同3.4 功能不同3.5 安全性不同3.6 性能不同3.7 价格不同3.8 两者可相互导入 4…

【大模型】提示工程基础学习

目录 1. 零样本提示2. 少样本提示3. 链式思考提示&#xff08;CoT&#xff09;4. 自我一致性5. 生成知识提示6. 链式提示7. 思维树&#xff08;ToT&#xff09;8. 检索增强生成&#xff08;RAG&#xff09;9. active prompt10. 自我反思&#xff08;reflexion&#xff09;11. 多…

儿童网络守护计划:如何为孩子营造一个纯净的在线空间?

青少年不想被监视&#xff0c;但他们需要受到保护&#xff0c;免受互联网危险。我们告诉您如何与您的孩子建立信任&#xff0c;同时了解他们在网上面临的挑战。 对于今天的孩子来说&#xff0c;将离线和在线朋友分开是不可能的。青少年在任何时候都与一切事物和每个人联系在一起…

2024年道路运输企业主要负责人证考试题库及道路运输企业主要负责人试题解析

题库来源&#xff1a;安全生产模拟考试一点通公众号小程序 2024年道路运输企业主要负责人证考试题库及道路运输企业主要负责人试题解析是安全生产模拟考试一点通结合&#xff08;安监局&#xff09;特种作业人员操作证考试大纲和&#xff08;质检局&#xff09;特种设备作业人…

MFC引用C#生成的dll,将dll放置到非exe程序目录,如何操作?

&#x1f3c6;本文收录于「Bug调优」专栏&#xff0c;主要记录项目实战过程中的Bug之前因后果及提供真实有效的解决方案&#xff0c;希望能够助你一臂之力&#xff0c;帮你早日登顶实现财富自由&#x1f680;&#xff1b;同时&#xff0c;欢迎大家关注&&收藏&&…

【Linux进阶】vim的用法

1.什么是vi/vim? 简单来说&#xff0c;vi是老式的文本编辑器&#xff0c;不过功能已经很齐全了&#xff0c;但是还是有可以进步的地方。vim则可以说是程序开发者的一项很好用的工具&#xff0c;就连 vim的官方网站&#xff08; http://www.vim.org&#xff09;自己也说vim是一…