引子 - 实现轻量的 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框架的性能,分两类 ...
随机推荐
- 前后端分离项目(九):实现"添加"功能(后端接口)
好家伙,来了来了,"查"已经完成了,现在是"增" 前端的视图已经做好了,现在我们来完善后端 后端目录结构 完整代码在前后端分离项目(五):数据分页查询(后端 ...
- Codeforces Round #812 (Div. 2) E(并查集)
种类并查集:定义种类之间的关系来判断操作是否进行 题目大意:对于题目给出的一个矩阵,我们可以进行一种操作:swap(a[i][j],a[j][i]) 使得矩阵可以变换为字典序最小的矩阵 思路: 通过扫 ...
- vue3的学习笔记:MVC、Vue3概要、模板、数据绑定、用Vue3 + element ui、react框架实现购物车案例
一.前端MVC概要 1.1.库与框架的区别 框架是一个软件的半成品,在全局范围内给了大的约束.库是工具,在单点上给我们提供功能.框架是依赖库的.Vue是框架而jQuery则是库. 1.2.MVC(Mo ...
- 新版的Eureka已经移除了基于Ribbon的客户端的负载均衡
启用一个EurekaServer和一个服务调用方,两个copy的服务提供方. 本次测试用Springcloud 2021.0.1版本 客户端使用RestTemplate 的负载均衡 @LoadBala ...
- 【接口测试】Postman(一)--接口测试知识准备
1.0 前言 应用程序编程接口(Application Programming Interface, API)是这些年来最流行的技术之一,强大的Web应用程序和领先的移动应用程序都离不开后端强大的 ...
- 社论 22.10.9 优化连续段dp
CF840C 给定一个序列 \(a\),长度为 \(n\).试求有多少 \(1\) 到 \(n\) 的排列 \(p_i\),满足对于任意的 \(2\le i\le n\) 有 \(a_{p_{i-1} ...
- 【大数据面试】【框架】Shuffle优化、内存参数配置、Yarn工作机制、调度器使用
三.MapReduce 1.Shuffle及其优化☆ Shuffle是Map方法之后,Reduce方法之前,混洗的过程 Map-->getPartition(标记数据的分区)-->对应的环 ...
- C++面向对象程序设计期末复习笔记[吉林大学](结合历年题速成85)
1.头文件 头文件的作用就是被其他的.cpp包含进去的.它们本身并不参与编译,但实际上,它们的内容却在多个.cpp文件中得到了编译.根据"定义只能一次"原则我们知道,头文件中不能放 ...
- 03.Javascript学习笔记2
1.逻辑运算符 在javascript中与或非对应的逻辑运算符是: && || ! const a = true; const b = false; console.log(a &am ...
- input、print、字符串格式化输出
1.使用input(), print()进行用户交互 """ 以前银行取钱只能拿着存折去柜台跟小姐姐交流才可以 你想干嘛 我想取钱 请输入密码 滴滴滴密码 想取多少钱 我 ...