前言:
最近忙里偷闲,写了该案例巩固下基础知识,如果不正之处敬请指出

控制反转: 将创建对象的权利交给别人 。
依赖注入: 控制反转的体现,容器在运行期间,动态地将某种依赖关系注入到对象之中。

思路: 扫描指定目录下的文件转换为Class集合,再通过反射进行实例化存入BeanMap中。

实现代码:

/*** @Author:wuruixu* @Date:2021/7/26*/
@Target({ElementType.FIELD})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface AutowiredAnnotation {}
/*** @Author:wuruixu* @Date:2021/7/26*/
@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
public @interface ControllerAnnotation {}
/*** @Author:wuruixu* @Date:2021/7/27* @Description: 控制反转及依赖注入实现工具*/
public class IocAndDiUtil {/*** 存储实例  (key: 全限定名   value:实例)*/public static ConcurrentHashMap<String, Object> beanMap = new ConcurrentHashMap<>();/*** 初始化BeanMap* @param packageName 包名 例: com.wook.demo.ioc_di* @throws Exception*/public static void initBeanMap(String packageName) throws Exception {//返回类上包含ControllerAnnotation注解的所有Class,并存进BeanMapSet<Class<?>> classSet = PackageUtil.listContainAnnotationClazz(packageName, true, ControllerAnnotation.class);for (Class<?> clazz : classSet) {String fullyName = clazz.getName();beanMap.put(fullyName, IocAndDiUtil.injectInstance(fullyName));}}/*** 为指定类中,带@AutowiredAnnotation 的成员变量注入实例  (可以理解为 DI依赖注入的实现)* @param fullyName 类的全限定名 例:com.xxx.xxx.xx.Controller* @return* @throws Exception*/public static Object injectInstance(String fullyName) throws Exception {Class<?> clazz = Class.forName(fullyName);Object obj = clazz.newInstance();Field[] declaredFields = clazz.getDeclaredFields();for (Field declaredField : declaredFields) {if (null != declaredField.getAnnotation(AutowiredAnnotation.class)) {declaredField.setAccessible(true);Class<?> fieldType = declaredField.getType();declaredField.set(obj, fieldType.newInstance());}}return obj;}
}
/*** @Author:wuruixu* @Date:2021/7/26* @Description: 名称空间实用工具*/
public final class PackageUtil {/*** 类默认构造器*/private PackageUtil() {}/*** 返回指定包中类上包含 annotationClazz 注解的Class对象** @param packageName 包名称* @param recursive   是否递归查找* @param annotationClazz 过滤条件: 类上应存在的注解class* @return 子类集合*/static public  Set<Class<?>> listContainAnnotationClazz(String packageName, boolean recursive, Class annotationClazz) {if (annotationClazz == null) {return Collections.emptySet();} else {return listClazz(packageName, recursive, clazz -> null != clazz.getAnnotation(annotationClazz));}}/*** 返回指定包中的所有子类Class对象** @param packageName 包名称* @param recursive   是否递归查找* @param superClazz  过滤条件: 属于该superClazz的子类* @return 子类集合*/static public Set<Class<?>> listSubClazz(String packageName, boolean recursive, Class<?> superClazz) {if (superClazz == null) {return Collections.emptySet();} else {//isAssignableFrom()方法是判断是否为某个类的父类,instanceof关键字是判断是否某个类的子类。return listClazz(packageName, recursive, superClazz::isAssignableFrom);}}/*** 列表指定包中的所有类** @param packageName 包名称* @param recursive   是否递归查找?* @param filter      过滤器* @return 符合条件的类集合*/static public Set<Class<?>> listClazz(String packageName, boolean recursive, IClazzFilter filter) {if (packageName == null || packageName.isEmpty()) {return null;}// 将点转换成斜杠final String packagePath = packageName.replace('.', '/');// 获取类加载器ClassLoader cl = Thread.currentThread().getContextClassLoader();// 结果集合Set<Class<?>> resultSet = new HashSet<>();try {// 获取 URL 枚举Enumeration<URL> urlEnum = cl.getResources(packagePath);while (urlEnum.hasMoreElements()) {// 获取当前 URLURL currUrl = urlEnum.nextElement();// 获取协议文本 (例 file 或 jar)final String protocol = currUrl.getProtocol();// 定义临时集合Set<Class<?>> tmpSet = null;if ("FILE".equalsIgnoreCase(protocol)) {// 从文件系统中加载类tmpSet = listClazzFromDir(new File(currUrl.getFile()), packageName, recursive, filter);} else if ("JAR".equalsIgnoreCase(protocol)) {// 获取文件字符串String fileStr = currUrl.getFile();if (fileStr.startsWith("file:")) {// 如果是以 "file:" 开头的,// 则去除这个开头fileStr = fileStr.substring(5);}if (fileStr.lastIndexOf('!') > 0) {// 如果有 '!' 字符,// 则截断 '!' 字符之后的所有字符fileStr = fileStr.substring(0, fileStr.lastIndexOf('!'));}// 从 JAR 文件中加载类tmpSet = listClazzFromJar(new File(fileStr), packageName, recursive, filter);}if (tmpSet != null) {// 如果类集合不为空,// 则添加到结果中resultSet.addAll(tmpSet);}}} catch (Exception ex) {// 抛出异常!throw new RuntimeException(ex);}return resultSet;}/*** 从目录中获取类列表** @param dirFile     目录* @param packageName 包名称* @param recursive   是否递归查询子包* @param filter      类过滤器* @return 符合条件的类集合*/static private Set<Class<?>> listClazzFromDir(final File dirFile, final String packageName, final boolean recursive, IClazzFilter filter) {if (!dirFile.exists() || !dirFile.isDirectory()) {// 如果参数对象为空,// 则直接退出!return null;}// 获取子文件列表File[] subFileArr = dirFile.listFiles();if (subFileArr == null || subFileArr.length <= 0) {return null;}// 文件队列, 将子文件列表添加到队列Queue<File> fileQ = new LinkedList<>(Arrays.asList(subFileArr));// 结果对象Set<Class<?>> resultSet = new HashSet<>();while (!fileQ.isEmpty()) {// 从队列中获取文件File currFile = fileQ.poll();if (currFile.isDirectory() && recursive) {// 如果当前文件是目录,// 并且是执行递归操作时,// 获取子文件列表subFileArr = currFile.listFiles();if (subFileArr != null && subFileArr.length > 0) {// 添加文件到队列fileQ.addAll(Arrays.asList(subFileArr));}continue;}if (!currFile.isFile() || !currFile.getName().endsWith(".class")) {// 如果当前文件不是文件,// 或者文件名不是以 .class 结尾,// 则直接跳过continue;}// 类名称String clazzName;// 设置类名称clazzName = currFile.getAbsolutePath();// 清除最后的 .class 结尾clazzName = clazzName.substring(dirFile.getAbsolutePath().length(), clazzName.lastIndexOf('.'));// 转换目录斜杠clazzName = clazzName.replace('\\', '/');// 清除开头的 /clazzName = trimLeft(clazzName, "/");// 将所有的 / 修改为 .clazzName = join(clazzName.split("/"), ".");// 包名 + 类名clazzName = packageName + "." + clazzName;try {// 加载类定义Class<?> clazzObj = Class.forName(clazzName);if (null != filter && !filter.accept(clazzObj)) {// 如果过滤器不为空,且过滤器不接受当前类,则直接跳过!continue;}// 添加类定义到集合resultSet.add(clazzObj);} catch (Exception ex) {// 抛出异常throw new RuntimeException(ex);}}return resultSet;}/*** 从 .jar 文件中获取类列表** @param jarFilePath .jar 文件路径* @param recursive   是否递归查询子包* @param filter      类过滤器* @return 符合条件的类集合*/static private Set<Class<?>> listClazzFromJar(final File jarFilePath, final String packageName, final boolean recursive, IClazzFilter filter) {if (jarFilePath == null || jarFilePath.isDirectory()) {// 如果参数对象为空,// 则直接退出!return null;}// 结果对象Set<Class<?>> resultSet = new HashSet<>();try {// 创建 .jar 文件读入流JarInputStream jarIn = new JarInputStream(new FileInputStream(jarFilePath));// 进入点JarEntry entry;while ((entry = jarIn.getNextJarEntry()) != null) {if (entry.isDirectory()) {continue;}// 获取进入点名称String entryName = entry.getName();if (!entryName.endsWith(".class")) {// 如果不是以 .class 结尾,// 则说明不是 JAVA 类文件, 直接跳过!continue;}if (!recursive) {//// 如果没有开启递归模式,// 那么就需要判断当前 .class 文件是否在指定目录下?// 获取目录名称String tmpStr = entryName.substring(0, entryName.lastIndexOf('/'));// 将目录中的 "/" 全部替换成 "."tmpStr = join(tmpStr.split("/"), ".");if (!packageName.equals(tmpStr)) {// 如果包名和目录名不相等,// 则直接跳过!continue;}}String clazzName;// 清除最后的 .class 结尾clazzName = entryName.substring(0, entryName.lastIndexOf('.'));// 将所有的 / 修改为 .clazzName = join(clazzName.split("/"), ".");// 加载类定义Class<?> clazzObj = Class.forName(clazzName);if (null != filter && !filter.accept(clazzObj)) {// 如果过滤器不为空,且过滤器不接受当前类,则直接跳过!continue;}// 添加类定义到集合resultSet.add(clazzObj);}// 关闭 jar 输入流jarIn.close();} catch (Exception ex) {// 抛出异常throw new RuntimeException(ex);}return resultSet;}/*** 类名称过滤器** @author hjj2019*/@FunctionalInterfacepublic interface IClazzFilter {/*** 是否接受当前类?** @param clazz 被筛选的类* @return 是否符合条件*/boolean accept(Class<?> clazz);}/*** 使用连接符连接字符串数组** @param strArr 字符串数组* @param conn   连接符* @return 连接后的字符串*/static private String join(String[] strArr, String conn) {if (null == strArr || strArr.length <= 0) {return "";}StringBuilder sb = new StringBuilder();for (int i = 0; i < strArr.length; i++) {if (i > 0) {// 添加连接符sb.append(conn);}// 添加字符串sb.append(strArr[i]);}return sb.toString();}/*** 清除源字符串左边的字符串** @param src     原字符串* @param trimStr 需要被清除的字符串* @return 清除后的字符串*/static private String trimLeft(String src, String trimStr) {if (null == src || src.isEmpty()) {return "";}if (null == trimStr || trimStr.isEmpty()) {return src;}if (src.equals(trimStr)) {return "";}while (src.startsWith(trimStr)) {src = src.substring(trimStr.length());}return src;}
}

准备测试数据:

/*** @Author:wuruixu* @Date:2021/7/26*/
@Data
@ControllerAnnotation
public class ControllerTest {@AutowiredAnnotationprivate ServiceOne serviceOne;@AutowiredAnnotationprivate ServiceTwo serviceTwo;
}class ServiceOne {public String fun1(){System.out.println("ServiceOne");return "ServiceOne";}
}class ServiceTwo {public String fun2(){System.out.println("ServiceTwo");return "ServiceTwo";}
}public static void main(String[] args) throws Exception {//初始化BeanIocAndDiUtil.initBeanMap("com.wook.demo.ioc_di");//测试实例是否成功注入ControllerTest controllerTest = (ControllerTest)IocAndDiUtil.beanMap.get("com.wook.demo.ioc_di.ControllerTest");controllerTest.getServiceOne().fun1();controllerTest.getServiceTwo().fun2();}

运行结果:
ServiceOne
ServiceTwo

学习样例: 模拟spring控制反转及依赖注入相关推荐

  1. Spring控制反转和依赖注入

    (1)为什么使用控制反转和依赖注入 调用者不用去关心被调用者的实现,不需要调用者亲自创建被调用者的实例,这些操作都交给框架去完成. 这种设计思想就是控制反转(Inversion of Control, ...

  2. Spring控制反转和依赖注入的好处

    要了解控制反转( Inversion of Control ), 我觉得有必要先了解软件设计的一个重要思想:依赖倒置原则(Dependency Inversion Principle ). 什么是依赖 ...

  3. Spring 控制反转和依赖注入简介

    文章目录 1.介绍 2.什么是控制反转? 3.什么是依赖注入? 4.Spring IoC 容器 5.基于构造函数的依赖注入 6.基于Setter的依赖注入 7.基于字段的依赖注入 8.自动装配依赖 9 ...

  4. Spring 控制反转和依赖注入

    控制反转的类型 控制反转(IOC)旨在提供一种更简单的机制,来设置组件的依赖项,并在整个生命周期管理这些依赖项.通常,控制反转可以分成两种子类型:依赖注入(DI)和依赖查找(DL),这些子类型各自又可 ...

  5. Spring控制反转(依赖注入)的最简单说明

    1.常规方式实现实例化 1.1已有角色如下: 一个接口Interface,两个接口实现类InstatnceA.InstanceB,一个调用类User 1.2当前实例化InstanceA如下: Inte ...

  6. spring控制反转与依赖注入

    1)什么是控制反转 传统的编程思路是,当我需要某个对象时,我便自己去实例化调用它 .而控制反转则是,当我需要某个对象时,自然有人帮我们实例化它 .简单的来说,这是一种衣来张口,饭来伸手式的控制模式,这 ...

  7. 控制反转 php,[PHP学习] 控制反转以及依赖注入的日常使用

    本篇文章小编将带大家学习一下PHP中的控制反转以及依赖注入的使用,感兴趣的小伙伴赶紧来看看吧! 控制反转:控制权交给了自己的类 依赖注入:依赖另一个类,我没有手动去new它 /*我自己要用的类*/ c ...

  8. 轻松了解Spring中的控制反转和依赖注入

    点击上方 "程序员小乐"关注公众号, 星标或置顶一起成长 每天早上8点20分, 第一时间与你相约 每日英文 When you have something you really l ...

  9. Spring容器,控制反转,依赖注入

    Spring boot学习之旅,为更好督促自己学习以记之,仅供参考. spring容器 程序启动的时候会创建spring容器,扫描给spring容器一个清单,比如:@Controller, @Bean ...

  10. Spring框架中的控制反转和依赖注入

    控制反转: 控制反转是用来降低代码之间的耦合度的,基本思想就是借助"第三方"实现具有依赖对象的解耦. 为什么需要控制反转,因为项目中对象或多或少存在耦合.控制反转的关键在于Ioc容 ...

最新文章

  1. Linux centos6.5 64下oracle10g_静默安装完整版
  2. Linux系统调用号表
  3. 我只是不甘心-------Day51
  4. 自动装箱自动拆箱java,自动装箱?拆箱?==问题?详解java面试常见的一个问题...
  5. 怎么实现java和数据库的链接_JAVA简单链接Oracle数据库,实现注册和登陆的功能
  6. Spring框架参考手册(4.2.6版本)翻译——第三部分 核心技术 6.10.6 给自动检测组件命名...
  7. 编程修养 阅读笔记二
  8. 合规不利于安全的五种情形
  9. 一个批量文件改名工具
  10. pyqt5菜鸟教程_PyQt5教程(一)——第一个PyQt5程序
  11. 未来五年你永远不会忘记的最佳GameCube仿真器
  12. wirelessn1000 驱动_Intel无线网卡驱动程序下载
  13. 秦始皇寻找长生药的始末
  14. RabbitMQ提示ERROR: epmd error for host
  15. 把Colab运行后生成的文件保存到Google drive
  16. 在CentOS 7配置IPv6 DNS Server
  17. 系统架构设计——伸缩性架构
  18. java责任链模式审批请假_14-学生生病请假:责任链模式
  19. Dubbo Cluster集群那点你不知道的事。
  20. JAVA开发技能要求:

热门文章

  1. RCS(1)随便写写没整理
  2. DHCP高阶应用系列之《dhcp-snooping + ipsg 拒绝非法DHCP服务器、拒绝用户随意修改IP地址》
  3. PD18 无法启动bootcamp,DlInitialzeLibrary failed 0xc00000bb的一个原因
  4. 天威诚信21周年丨以梦为马 不负韶华
  5. 怎样才能提高异步电动机自然功率因数
  6. python get hist data_[宜配屋]听图阁
  7. Chromium GN 参照(chromium GN 构建系统学习参考)
  8. 微信与服务器通信失败,微信公众平台开发服务器与微信服务器通信时失败 获取不到值...
  9. 斯坦福大学cs229学习体会(1)-机器学习入门
  10. 申宝策略-观望为主等待市场转强