手写实现简单版IOC
概述
IOC (Inversion of Control) 控制反转,大家应该都比较熟悉了、应该也都有用过,这里就不具体介绍了。自己平时也有用到过IOC,但是对它的具体实现原理只有一个模糊的概念,所以决定自己手动实现一个简单IOC。
开始
首先呢我们得知道IOC的主要作用是什么,才能开始动手写。IOC主要不就是负责创建对象以及管理生命周期嘛,那我们就开始动手啦。
比如现在有一个IAnimal接口Animal继承接口,然后就是个Call的方法。一般我们使用的时候都是IAnimal animal=new Animal(); 如果是使用第三方IOC容器实现的话,我们需要先注册一下类型才能获取到实例。
所以我们先来个最简单的仿照这个过程:
新建一个Container,然后里面有一个类型注册的方法ResgisterType和一个返回实例的方法Rerolve,还有一个存储类型的字典,具体代码如下
private static Dictionary<string, object> ContainerTypeDictionary = new Dictionary<string, object>();/// <summary>
/// 注册类型
/// </summary>
/// <typeparam name="IT"></typeparam>
/// <typeparam name="T"></typeparam>
public void ResgisterType<IT,T>()
{
if (!ContainerTypeDictionary.ContainsKey(typeof(IT).FullName))
ContainerTypeDictionary.Add(typeof(IT).FullName, typeof(T));
} /// <summary>
/// 根据注册信息生成实例
/// </summary>
/// <typeparam name="IT"></typeparam>
/// <returns></returns>
public IT Rerolve<IT>()
{
string key = typeof(IT).FullName;
Type type = (Type)ContainerTypeDictionary[key];
return (IT)Activator.CreateInstance(type);
}
然后我们新建一个控制台测试一下
Container container = new Container();
container.ResgisterType<IAnimal, Animal>();
IAnimal animal= container.Rerolve<IAnimal>();
然后可以在不依赖具体对象Animal的情况下成功的创建一个animal实例。
之后我们就可以考虑复杂一点的情况了,现在我们的Animal类里没有做任何事,假如它的构造函数里依赖于另一个对象呢,这样我们的程序肯定是会报错的。比如下面这样:
public class Animal: IAnimal
{
public Animal(Dog dog)
{ }
}
我们容器目前能创建的对象实例,只有通过ResgisterType方法注册过类型的,而像Animal里依赖的不能实现创建,所以这个时候就需要用到依赖注入了。
关于依赖注入与控制反转的关系,我个人的理解是:控制反转是一种设计思想,而依赖注入则是实现控制反转思想的方法。
IOC容器一般依赖注入有三种:构造函数注入、方法注入、属性注入。
那么我们就来照瓢画葫芦,实现一下构造函数注入。一般IOC容器构造函数注入是通过一个特性来识别注入的,如果没有标记特性则去找构造函数参数个数最多的,我们就按照这个思路来。
首先我们新建一个LInjectionConstructorAttribute类,只需继承Attribute就行了。
public class LInjectionConstructorAttribute :Attribute
{
}
然后在刚才那个Animal构造函数上标记上特性,接下来就开始写代码。
/// <summary>
/// 根据注册信息生成实例
/// </summary>
/// <typeparam name="IT"></typeparam>
/// <returns></returns>
public IT Rerolve<IT>()
{
string key = typeof(IT).FullName;
Type type = (Type)ContainerTypeDictionary[key]; return (IT)CreateType(type);
}
/// <summary>
/// 根据提供的类型创建类型实例并返回
/// </summary>
/// <param name="type"></param>
/// <returns></returns>
private object CreateType(Type type)
{
var ctorArray = type.GetConstructors();
if (ctorArray.Count(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)) > )
{
//获取带特性标记的构造函数参数
foreach (var cotr in type.GetConstructors().Where(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)))
{
var paraArray = cotr.GetParameters();//获取参数数组
if (paraArray.Length == )
{
return Activator.CreateInstance(type);
} List<object> listPara = new List<object>();
foreach (var para in paraArray)
{
string paraKey = para.ParameterType.FullName;//参数类型名称
//从字典中取出缓存的目标对象并创建对象
Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
object oPara = CreateType(paraTargetType);//递归
listPara.Add(oPara);
}
return Activator.CreateInstance(type,listPara.ToArray());
} return Activator.CreateInstance(type);
}
else
{
//没有标记特性则使用参数最多的构造函数
var ctor = ctorArray.OrderByDescending(c => c.GetParameters().Length).FirstOrDefault();
var paraArray = ctor.GetParameters();//获取参数数组
if (paraArray.Length == )
{
return Activator.CreateInstance(type);
} List<object> listPara = new List<object>();
foreach (var para in paraArray)
{
string paraKey = para.ParameterType.FullName;//参数类型名称
//从字典中取出缓存的目标对象并创建对象
Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
object oPara = CreateType(paraTargetType);//递归
listPara.Add(oPara);
}
return Activator.CreateInstance(type, listPara.ToArray());
}
}
这里说下为什么用到递归,在我们项目中使用会有层层依赖的关系。比如我这里Animal依赖于Dog只有一层依赖,如果Gog又依赖于猫、猫依赖于鱼。。。(当然这里只是打个比方)
因为我们不知道具体有几层依赖,所以使用了递归的方法,直到将所有依赖的对象得到后再创建实例。
然后我们再来测试
Container container = new Container();
container.ResgisterType<IAnimal, Animal>();
container.ResgisterType<IDog, Dog>();
IAnimal animal= container.Rerolve<IAnimal>();
注意,如果测试标记特性的一定不要忘了在构造函数上标记特性,然后我们会发现最终也可以得到animal对象。
然后,创建对象这一块我们先告一段落。接下来进行生命周期管理。
一般的IOC容器都支持三种类型:Transient每次都得到一个新的对象、Scoped同一个域(或者请求、线程)中使用同一个对象、Singleton整个程序生命周期都使用同一实例对象。
那按照我们以上的代码怎么才能实现生命周期管理呢?我是这么想的:既然创建对象的工作都是由我容器来做了,那么我们在创建完对象之后能不能像注册类型一样将对象保存起来呢?
所以我这里使用了简单的字典来存储对象实例,然后通过判断使用的哪一种生命周期来返回新的对象或是直接返回字典里的对象。直接改造上面的代码了:
/// <summary>
/// 根据提供的类型创建类型实例并返回
/// </summary>
/// <param name="type"></param>
/// <returns></returns>
private object CreateType(Type type)
{
var ctorArray = type.GetConstructors();
if (ctorArray.Count(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)) > )
{
//获取带特性标记的构造函数参数
foreach (var cotr in type.GetConstructors().Where(c => c.IsDefined(typeof(LInjectionConstructorAttribute), true)))
{
var paraArray = cotr.GetParameters();//获取参数数组
if (paraArray.Length == )
{
//return Activator.CreateInstance(type);
return GetSocpe(type);
} List<object> listPara = new List<object>();
foreach (var para in paraArray)
{
string paraKey = para.ParameterType.FullName;//参数类型名称
//从字典中取出缓存的目标对象并创建对象
Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
object oPara = CreateType(paraTargetType);//递归
listPara.Add(oPara);
}
//return Activator.CreateInstance(type,listPara.ToArray());
return GetSocpe(type, listPara.ToArray());
} return GetSocpe(type);
//return Activator.CreateInstance(type);
}
else
{
//没有标记特性则使用参数最多的构造函数
var ctor = ctorArray.OrderByDescending(c => c.GetParameters().Length).FirstOrDefault();
var paraArray = ctor.GetParameters();//获取参数数组
if (paraArray.Length == )
{
//return Activator.CreateInstance(type);
return GetSocpe(type);
} List<object> listPara = new List<object>();
foreach (var para in paraArray)
{
string paraKey = para.ParameterType.FullName;//参数类型名称
//从字典中取出缓存的目标对象并创建对象
Type paraTargetType = (Type)ContainerTypeDictionary[paraKey];
object oPara = CreateType(paraTargetType);//递归
listPara.Add(oPara);
}
return GetSocpe(type, listPara.ToArray());
//return Activator.CreateInstance(type, listPara.ToArray());
}
}
private object GetSocpe(Type type, params object[] listPara)
{
if (_scopeType == (int)Scope.Singleton)
{
return GetTypeSingleton(type, listPara);
}
else if (_scopeType == (int)Scope.Transient)
{
return GetTypeTransient(type, listPara);
}
else
{
return GetTypeScoped(type, listPara);
}
} #region 生命周期
/// <summary>
/// 设置获取实例对象生命周期为Singleton
/// </summary>
/// <param name="type"></param>
/// <param name="listPara"></param>
/// <returns></returns>
private object GetTypeSingleton(Type type, params object[] listPara)
{
if (ContainerExampleDictionary.ContainsKey(type.FullName))
{
lock (locker)
{
if (ContainerExampleDictionary.ContainsKey(type.FullName))
{
return ContainerExampleDictionary[type.FullName];
}
}
} if (listPara.Length == )
{
var Example = Activator.CreateInstance(type);
ContainerExampleDictionary.Add(type.FullName, Example);
return Example;
}
else
{
var Example = Activator.CreateInstance(type, listPara.ToArray());
ContainerExampleDictionary.Add(type.FullName, Example);
return Example;
}
} /// <summary>
/// 设置获取实例对象生命周期为Transient
/// </summary>
/// <param name="type"></param>
/// <param name="listPara"></param>
/// <returns></returns>
private object GetTypeTransient(Type type, params object[] listPara)
{
if (listPara.Length == )
{
var Example = Activator.CreateInstance(type);
//ContainerExampleDictionary.Add(type.FullName, Example);
return Example;
}
else
{
var Example = Activator.CreateInstance(type, listPara.ToArray());
//ContainerExampleDictionary.Add(type.FullName, Example);
return Example;
}
} /// <summary>
/// 设置获取实例对象生命周期为Scoped
/// </summary>
/// <param name="type"></param>
/// <param name="listPara"></param>
/// <returns></returns>
private object GetTypeScoped(Type type, params object[] listPara)
{
var pid = System.Threading.Thread.CurrentThread.ManagedThreadId;
if (ContainerExampleDictionary.ContainsKey(type.FullName + pid))
{
lock (locker)
{
if (ContainerExampleDictionary.ContainsKey(type.FullName + pid))
{
return ContainerExampleDictionary[type.FullName + pid];
}
}
} if (listPara.Length == )
{
var Example = Activator.CreateInstance(type);
ContainerExampleDictionary.Add(type.FullName + pid, Example);
return Example;
}
else
{
var Example = Activator.CreateInstance(type, listPara.ToArray());
ContainerExampleDictionary.Add(type.FullName + pid, Example);
return Example;
}
}
#endregion
private static Dictionary<string, object> ContainerExampleDictionary = new Dictionary<string, object>();
private static int _scopeType;
private static readonly object locker = new object();
public int scopeType
{
get
{
return _scopeType;
}
set
{
_scopeType = value;
}
}
public enum Scope
{
Singleton = ,
Transient = ,
Scoped =
}
然后调用的时候先声明下要使用的声明周期类型就行啦
Container container = new Container();
container.scopeType = (int)Container.Scope.Singleton;
container.ResgisterType<IAnimal, Animal>();
container.ResgisterType<IDog, Dog>();
IAnimal animal= container.Rerolve<IAnimal>();
说下三种生命周期管理的实现:
Transient:则可以直接创建一个实例
Scoped:使用的是同一个线程内使用同一个对象实例,使用var pid = System.Threading.Thread.CurrentThread.ManagedThreadId;获取线程id来判断的
Singleton:这种则只需一个单例模式获取就好了
到这里就先告一段落了,以上只是一个简单实现,代码还有需改进的地方以及可以扩展的功能,欢迎提意见指出错误。同时代码已上传GigHub,还有不懂的可以参考下代码。
源码地址:https://github.com/liangchengxuyuan/IocContainer
手写实现简单版IOC的更多相关文章
- 手写一个简单版的SpringMVC
一 写在前面 这是自己实现一个简单的具有SpringMVC功能的小Demo,主要实现效果是; 自己定义的实现效果是通过浏览器地址传一个name参数,打印“my name is”+name参数.不使用S ...
- 动手写一个简单版的谷歌TPU-矩阵乘法和卷积
谷歌TPU是一个设计良好的矩阵计算加速单元,可以很好的加速神经网络的计算.本系列文章将利用公开的TPU V1相关资料,对其进行一定的简化.推测和修改,来实际编写一个简单版本的谷歌TPU.计划实现到行为 ...
- laravel学习:php写一个简单的ioc服务管理容器
php写一个简单的ioc服务管理容器 原创: 陈晨 CoderStory 2018-01-14 最近学习laravel框架,了解到laravel核心是一个大容器,这个容器负责几乎所有服务组件的实例化以 ...
- 动手写一个简单版的谷歌TPU-指令集
系列目录 谷歌TPU概述和简化 基本单元-矩阵乘法阵列 基本单元-归一化和池化(待发布) TPU中的指令集 SimpleTPU实例: (计划中) 拓展 TPU的边界(规划中) 重新审视深度神经网络中的 ...
- 利用SpringBoot+Logback手写一个简单的链路追踪
目录 一.实现原理 二.代码实战 三.测试 最近线上排查问题时候,发现请求太多导致日志错综复杂,没办法把用户在一次或多次请求的日志关联在一起,所以就利用SpringBoot+Logback手写了一个简 ...
- 手写一个简单的starter组件
spring-boot中有很多第三方包,都封装成starter组件,在maven中引用后,启动springBoot项目时会自动装配到spring ioc容器中. 思考: 为什么我们springBoot ...
- 手写一个简单的ElasticSearch SQL转换器(一)
一.前言 之前有个需求,是使ElasticSearch支持使用SQL进行简单查询,较新版本的ES已经支持该特性(不过貌似还是实验性质的?) ,而且git上也有elasticsearch-sql 插件, ...
- 识别手写数字增强版100% - pytorch从入门到入道(一)
手写数字识别,神经网络领域的“hello world”例子,通过pytorch一步步构建,通过训练与调整,达到“100%”准确率 1.快速开始 1.1 定义神经网络类,继承torch.nn.Modul ...
- 手写一个简单到SpirngMVC框架
spring对于java程序员来说,无疑就是吃饭到筷子.在每次编程工作到时候,我们几乎都离不开它,相信无论过去,还是现在或是未来到一段时间,它仍会扮演着重要到角色.自己对spring有一定的自我见解, ...
随机推荐
- 工作流调度器azkaban
为什么需要工作流调度系统 一个完整的数据分析系统通常都是由大量任务单元组成: shell脚本程序,java程序,mapreduce程序.hive脚本等 各任务单元之间存在时间先后及前后依赖关系 为了很 ...
- leetcode1027
最直接的思路是三层循环,但是会超时,代码如下: public class Solution { public int LongestArithSeqLength2(int[] A) { ; var l ...
- poi横纵导出
dao <select id="selectTargetModel" resultMap="targetMap"> select si.SHOP_N ...
- 关于div文字点击编辑的插件
(function(w,i){ w.inputOut = new i(); })( window, function(){ var inputOut = function(){ this.into = ...
- BASIC GIT WORKFLOW
BASIC GIT WORKFLOW Generalizations You have now been introduced to the fundamental Git workflow. You ...
- GDI+_Png图片浏览器
'昨天看见有人问VB6支不支持PNG,刚好我正在研究GDI+,于是做了这个演示程序.'演示下载地址:百度网盘|'下面为设计界面和运行效果截图 ' 千万别喷界面丑. /无奈 .
- DML DDL DCL
转自:https://blog.csdn.net/level_level/article/details/4248685
- Java的学习05
今天学习了,Java中的LinkedList类.这个类需要用到链表的知识,以前一直以为,只有c/c++有链表.今天才知道,原来其他语言.也有链表,而且还是双向链表. /** * 自定义一个链表 * @ ...
- DJango 基础 (4)
Django模板标签 知识点: 基本概念 常用标签 模板标签例子 模板继承与应用 注释标签 模板标签 标签在渲染的过程中提供任意的逻辑. 这个定义是刻意模糊的. 例如,一个标签可以输出内容,作为控制结 ...
- 【转载】一个小时学会MySQL数据库
一个小时学会MySQL数据库 目录 一.数据库概要 1.1.发展历史 1.1.1.人工处理阶段 1.1.2.文件系统 1.1.3.数据库管理系统 1.2.常见数据库技术品牌.服务与架构 1.3.数 ...