引子 - 实现轻量的 ioc 容器
IoC 反转控制原则也被叫做依赖注入 DI, 容器按照配置注入实例化的对象.
假设 A 的相互依赖关系如下图, 如何将 A 对象实例化并注入属性.
本文将实现一个轻量化的 IoC 容器, 完成对象的实例化和注入, 基于注解不依赖于任何库. (注解参考 JSR-330)
前提 JSR-330
| 注解 | 说明 |
|---|---|
| @Inject | 标识可注入的字段 |
| @Named | 基于字符串的限定符, 表示需要 IoC 接管的类 |
JSR-330 远比前提中提到的更多, 可以看下官方的解释说明, 这里只截取了本文目的需要开发的部分.
类定义
按照背景中的依赖关系图, 先定义出来对象.
@Named("a")
public class A {
@Inject
public B b;
@Inject
public C c;
// getter and setter
// constructor
}
@Named("b")
public class B {
@Value("hello world!")
public String name;
// getter and setter
// constructor
}
@Named("c")
public class C {
@Inject
public A a;
// getter and setter
// constructor
}
为了清晰, 这里省略了构造器和 setter 函数, 这些对于实现是必要的, 如果需要完整代码可以参照项目 xnuc-insni.

先考虑简单情况, A 与 B 的相互依赖如何实现.
注解定义
注解定义参照 inject, 这里只截取了需要的部分.
@Target(FIELD)
@Retention(RUNTIME)
public @interface Inject {}
@Target(TYPE)
@Retention(RUNTIME)
public @interface Named {
String value() default "";
}
对于简单类型, 可以提供一个设定的数值, 使用 Val 注解完成.
@Target(FIELD)
@Retention(RUNTIME)
public @interface Value {
String value() default "";
}
容器定义
容器定义很简单, 有一个实例的表和类定义的表.
public class Context {
public HashMap<String, Object> instances; // 实例
public HashMap<String, Class<?>> defineds; // 类定义
}
获取类定义
获取类定义用到反射和注解, 不了解相关知识的同学可以先补一下这部分. 如果要获取类定义, 最简单的方法就是找到全部类进行类加载. 首先获取主类加载器, 找到全部 .class 路径.
Enumeration<URL> resources = Main.class.getClassLoader().getResources(pkg.replace(".", "/"));
File file = new File(resources.nextElement().getFile());
获取全部包下的全部类, 存在子包的情况, 可以用递归或者队列, 最开始用的队列, 但是发现队列对于子包处理时非常复杂的, 需要根据队列信息维护当前包名. 递归的系统栈会帮我们记录下来自然就不需要我们自己维护了, 选择递归的方式处理子包.
private void subdir(String pkg, File file, List<Class<?>> clzes) throws Exception {
for (File f : file.listFiles()) { // 退出条件
if (f.isFile()) {
String clsName = String.format("%s.%s", pkg, f.getName().substring(0, f.getName().lastIndexOf(".")));
clzes.add(Class.forName(clsName));
}
if (f.isDirectory())
subdir(String.format("%s.%s", pkg, f.getName()), f, clzes);
}
}
这里选择了参数传返回值, 更好的方式还是直接内部将 list 构造出来, 返回出去.
拿到全部类后, 将有存在注解的类筛选出来. 放入 Context 的 defineds.
for (Class<?> c : clzList) {
if (c.getAnnotation(Named.class) != null) {
defineds.put(c.getAnnotation(Named.class).value(), c);
}
}
此时初始化完毕, 类定义获取到. 另外, 其实这里已经可以开始注入了, 但是真实情况下, 如果类定义比较多, 那么初始化将非常耗时, 如果选择用到再说的原则, 初始化就会快很.
用到再说
Context#get 用来获取容器对象, 如果对象还没有实例化, 就实例化. 实例化 instance 实现比较简单, 找到定义的 filed 进行 set. 这也解释了, 为什么需要无参构造器和 setter. 对于基础值也可以通过 @Value 主动赋予自定义的数值. 对于 @Inject 直接去 Context#get即可.
private Object instance(Object rto, Class<?> clz) throws Exception {
for (Field field : clz.getFields()) {
if (field.getAnnotation(Value.class) != null) {
PropertyDescriptor descriptor = new PropertyDescriptor(field.getName(), clz);
descriptor.getWriteMethod().invoke(rto, field.getAnnotation(Value.class).value());
}
if (field.getAnnotation(Inject.class) != null) {
PropertyDescriptor descriptor = new PropertyDescriptor(field.getName(), clz);
descriptor.getWriteMethod().invoke(rto, get(field.getName()));
}
}
return rto;
}
写完 instance 的逻辑, get 的逻辑就比较简单了. 有返回, 没有实例化.
public Object get(String objName) throws Exception {
if (instances.get(objName) != null)
return instances.get(objName);
if (defineds.get(objName) == null)
throw new Exception(String.format("objName %s undefined", objName));
Class<?> clz = defineds.get(objName);
instances.put(objName, instance(unreadyInstances.get(objName), clz));
return instances.get(objName);
}
这样只能解决 A 和 B 的问题, 对于 A 和 C 的问题这样就会导致注入 A 时发现需要注入 C, 而注入 C 时又要去注入 A, 最终导致循环.
依赖循环
循环依赖解决方法很简单, 只需要一个表记录下我现在正在注入 A, 所以 C 注入 A 的时候直接把正在注入的 A 丢给 C 即可.
所以新增 Context 成员 public HashMap<String, Object> unreadyInstances
public class Context {
public HashMap<String, Object> instances;
++ public HashMap<String, Object> unreadyInstances;
public HashMap<String, Class<?>> defineds;
}
注入前先把这个对象扔进去, 注入时如果其他对象有循环依赖, Context#get 可以直接返回这个对象.
public Object get(String objName) throws Exception {
if (instances.get(objName) != null)
return instances.get(objName);
++ if (unreadyInstances.get(objName) != null)
++ return unreadyInstances.get(objName);
if (defineds.get(objName) == null)
throw new Exception(String.format("objName %s undefined", objName));
Class<?> clz = defineds.get(objName);
++ unreadyInstances.put(objName, clz.getDeclaredConstructor().newInstance());
instances.put(objName, instance(unreadyInstances.get(objName), clz));
++ unreadyInstances.remove(objName);
return instances.get(objName);
}
最终的代码就是这样了, 写个 Main 类, 运行下.
public class Main {
public static void main(String[] args) throws Exception {
Context ioc = Context.run(Main.class);
A a = (A) ioc.get("a");
System.out.println(a.getC().getA().getB().getName()); // >hello world!
}
}
全部代码可以参考 xnuc - insni 喜欢可以帮忙点个 Star.
引子 - 实现轻量的 ioc 容器的更多相关文章
- 曹工说Tomcat4:利用 Digester 手撸一个轻量的 Spring IOC容器
一.前言 一共8个类,撸一个IOC容器.当然,我们是很轻量级的,但能够满足基本需求.想想典型的 Spring 项目,是不是就是各种Service/DAO/Controller,大家互相注入,就组装成了 ...
- 通过中看不中用的代码分析Ioc容器,依赖注入....
/** * 通过生产拥有超能力的超人实例 来理解IOC容器 */ //超能力模组接口 interface SuperModuleInterface{ public function activate( ...
- IoC容器Autofac - Autofac + Asp.net MVC + EF Code First(转载)
转载地址:http://www.cnblogs.com/JustRun1983/archive/2013/03/28/2981645.html 有修改 Autofac通过Controller默认构造 ...
- 比Spring简单的IoC容器
比Spring简单的IoC容器 Spring 虽然比起EJB轻量了许多,但是因为它需要兼容许多不同的类库,导致现在Spring还是相当的庞大的,动不动就上40MB的jar包, 而且想要理解Spring ...
- laravel5.8 IoC 容器
网上 对容器的解释有很多,这里只是记录,搬运! 1.简单理解: 2019-10-10 11:24:09 解析 lavarel 容器 IoC 容器 作用 就是 “解耦” .“依赖注入(DI) IoC 容 ...
- 【曹工杂谈】Maven IOC容器的下半场:Google Guice
Maven容器的下半场:Guice 前言 在前面的文章里,Maven底层容器Plexus Container的前世今生,一代芳华终落幕,我们提到,在Plexus Container退任后,取而代之的底 ...
- 【最简单IOC容器实现】实现一个最简单的IOC容器
前面DebugLZQ的两篇博文: 浅谈IOC--说清楚IOC是什么 IoC Container Benchmark - Performance comparison 在浅谈IOC--说清楚IOC是什么 ...
- 挖坟之Spring.NET IOC容器初始化
因查找ht项目中一个久未解决spring内部异常,翻了一段时间源码.以此文总结springIOC,容器初始化过程. 语言背景是C#.网上有一些基于java的spring源码分析文档,大而乱,乱而不全, ...
- SqlSugar轻量ORM
蓝灯软件数据股份有限公司项目,代码开源. SqlSugar是一款轻量级的MSSQL ORM ,除了具有媲美ADO的性能外还具有和EF相似简单易用的语法. 学习列表 0.功能更新 1.SqlSuga ...
- IOC容器MEF在MVC中的使用
最近想把自己的网站框架用IOC改造下,经过对比,我初步选择autofac,虽然MEF不需要配置,但性能不行,autofac虽然需要自己写自动化注入,但性能非常好. 先分析下各大IOC框架的性能,分两类 ...
随机推荐
- Unity——第一人称控制器的实现
Unity--第一人称控制器的实现 一.功能描述 在一个场景中实现人物的前后左右移动和跳跃功能:其中前后左右移动通过W.A.S.D方向键实现,跳跃功能通过空格键实现,并且考虑到重力作用,来调节跳跃功能 ...
- 类的编写模板之简单Java类
简单Java类是初学java时的一个重要的类模型,一般由属性和getter.setter方法组成,该类不涉及复杂的逻辑运算,仅仅是作为数据的储存,同时该类一般都有明确的实物类型.如:定义一个雇员的类, ...
- Python基础部分:12、文件光标移动(补充)
目录 一.文件内光标移动实际案例 二.计算机硬盘修改数据的原理 三.文件内容修改 一.文件内光标移动实际案例 # 1.二进制,只读模式,打a.txt文件 with open(r'a.txt', 'rb ...
- Terminal(oh-my-zsh) 美化
如果你使用Mac进行开发,那么Terminal.app应该是你使用非常频繁的app了.初体验Terminal时你可能觉得单调乏味,阅读密密麻麻的内容也很费劲.但是如果你跟着我一起配置它,就会发现你平时 ...
- Rust构建环境搭建
###安装涉及的概念rustup : 安装rust和管理版本的工具,当前rust尚处于发展阶段,存在三种类型的版本,稳定版.测试版.每日构建版本,使用rustup可以在这三种的版本之间切换,默认是稳定 ...
- ArrayList 可以完全替代数组吗?
本文已收录到 GitHub · AndroidFamily,有 Android 进阶知识体系,欢迎 Star.技术和职场问题,请关注公众号 [彭旭锐] 加入 Android 交流群. 前言 大家好, ...
- 黏包现象、struct模块和解决黏包问题的流程、UDP协议、并发编程理论、多道程序设计技术及进程理论 _
目录 黏包现象 二.struct模块及解决黏包问题的流程 三.粘包代码实战 UDP协议(了解) 并发编程理论 多道技术 进程理论 进程的并行与并发 进程的三状态 黏包现象 什么是粘包 1.服务端连续执 ...
- 再见CMS
观察网站最下方,根据备案号搜到这是个齐博CMS,然后百度就可以搜到齐博CMS漏洞了 然后开始利用 首先尝试了在用户信息修改处进行注入,发现好像想不通了,就在博客界面进行注入 Payload: 获取版本 ...
- 解决linux vlc设置中文问题
解决方法 sudo apt install vlc-l10n
- Sqoop的介绍和安装
sqoop下载地址:https://share.weiyun.com/woVbfnfS 或者 https://archive.apache.org/dist/sqoop/1.99.7/ Sqoop简介 ...