依赖注入(IOC)二
依赖注入(IOC)二
上一章我们讲了构造注入与设值注入,这一篇我们主要讲接口注入与特性注入。
接口注入
接口注入是将抽象类型的入口以方法定义在一个接口中,如果客户类型需要获得这个方法,就需要以实现这个接口的方式完成注入。实际上接口注入有很强的侵入性,除了要求客户类型增加前面两种方式所需要的代码外,还必须显示地定义一个新的接口并要求客户类型实现它。
//定义需要注入ITimeProvider的类型
interface IobjectWithTimeProvider
{
ITimeProvider TimeProvider { get; set; }
} //通过接口方式实现注入
public class Client:IobjectWithTimeProvider
{
public ITimeProvider TimeProvider { get; set; }
}
Unit Test
[TestClass]
public class TestClent
{
[TestMethod]
public void TestMethod1()
{
ITimeProvider timeProvider =
(new Assembler()).Create<ITimeProvider>(); Assert.IsNotNull(timeProvider);//确认可以正常获得抽象类型实例 IObjectWithTimeProvider objectWithProvider = new Client();
objectWithProvider.TimeProvider = timeProvider;//通过接口方式注入 }
}
随着C#语言的发展,接口注入可以采用与设值注入方式相似的方式实现,而且看上去很“Lamada化”。因为不用真正去实现接口,而是通过泛型参数的方式实现,可以说泛型为C#实现接口注入提供了“新生”。
/// <summary>
/// 通过泛型参数实现接口注入
/// </summary>
/// <typeparam name="T">待注入的接口类型</typeparam>
public class Client<T>:ITimeProvider
where T:ITimeProvider
{
/// <summary>
/// 与设值方式相似的注入入口
/// </summary>
public T Provider { get; set; } /// <summary>
/// 类似传统接口注入的实现方式
/// </summary>
public DateTime CurrentDate
{
get { return Provider.CurrentDate; }
}
}
Unit Test
[TestMethod]
public void Test()
{
var clietn = new Client<ITimeProvider>()
{
Provider = (new Assembler().Create<ITimeProvider>())
}; //验证设置方式注入的内容
Assert.IsNotNull(clietn.Provider);
Assert.IsNotInstanceOfType(clietn.Provider, typeof(SystemTimeProvider)); //验证注入的接口是否可用
Assert.IsNotInstanceOfType(clietn.Provider.CurrentDate, typeof(DateTime)); //验证是否满足传统接口注入的要求
Assert.IsTrue(typeof(ITimeProvider).IsAssignableFrom(clietn.GetType()));
}
基于特性的注入方式(Attributer)
直观上,客户程序可能在使用上做出让步以适应变化,但这违背了依赖注入的初衷,即三个角色(客户对象、Assembler、抽象类型)之中两个不能变,如果在Assembler和客户类型选择,为了客户对象影响最小,我们只好在Assembler上下功夫,因为它的职责是负责组装。反过来讲,如果注入过程还需要修改客户程序,那我们就没有必要去“削足适履”地去用“依赖注入”了。
因此,为了能通过特性方式完成依赖注入,我们只好在Assembler上下功夫
(错误的实现情况)
class SystemTimeAttribute:Attribute,ITimeProvider{…}
[SystemTime]
class Client{…}
相信读者也发现了,这样做虽然把客户类型需要的ITimeProvider通过“贴标签”的方式告诉它了,但事实上又把客户程序与SystemTimeAttribute“绑”上了,他们紧密的耦合在一起了。参考上面的三个实现,当抽象类型与客户对象耦合的时候我们就要用Assembler解耦。
当特性方式出现类似情况时,我们写一个AtttibuteAssembler不就行了吗?
还不行,设计上要把Attribute设计成一个通道,出于扩展和通用性的考虑,它本身要协助AtttibuteAssembler完成ITimeProvider的装配,最好还可以同时装载其他抽象类型来修饰客户类型。
示例代码如下
[AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public sealed class DecoratorAttribute : Attribute
{
//实现客户类型实际需要的抽象类型的实体类型实例,即待注入客户类型的内容
public readonly object Injector;
readonly Type type; public DecoratorAttribute(Type type)
{
if (type == null) throw new ArgumentNullException("type");
this.type = type; Injector = (new Assembler()).Create(this.type);
} //客户类型需要的抽象对象类型
public Type Type { get { return this.type; } }
} public static class AttributeHelper
{
public static T Injector<T>(object target) where T : class
{
if (target == null) throw new ArgumentNullException("target");
return (T)(((DecoratorAttribute[])
target.GetType().GetCustomAttributes(typeof(DecoratorAttribute), false))
.Where(x => x.Type == typeof(T)).FirstOrDefault()
.Injector);
}
}
[Decorator(typeof(ITimeProvider))]
//应用Attribute,定义需要将ITimeProvider通过它注入
class Client
{
public int GetYear()
{
//与其他注入不同的是,这里使用ITimeProvider来自自己的Attribute
var porvider = AttributeHelper.Injector<ITimeProvider>(this);
return porvider.CurrentDate.Year;
}
}
Unit Test
[TestMethod]
public void Test1()
{
Assert.IsTrue(new Client().GetYear() > 0);
}
依赖注入(IOC)二的更多相关文章
- android使用篇(四) 注解依赖注入IOC实现绑定控件
在android使用篇(三) MVC模式中提到一个问题: 1) 视图层(View):一般採用XML文件进行界面的描写叙述,使用的时候能够很方便的引入,可是用xml编写了,又须要在Acitvity声明而 ...
- 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之九 || 依赖注入IoC学习 + AOP界面编程初探
更新 1.如果看不懂本文,或者比较困难,先别着急问问题,我单写了一个关于依赖注入的小Demo,可以下载看看,多思考思考注入的原理: https://github.com/anjoy8/BlogArti ...
- Z从壹开始前后端分离【 .NET Core2.2/3.0 +Vue2.0 】框架之九 || 依赖注入IoC学习 + AOP界面编程初探
本文梯子 本文3.0版本文章 更新 代码已上传Github+Gitee,文末有地址 零.今天完成的绿色部分 一.依赖注入的理解和思考 二.常见的IoC框架有哪些 1.Autofac+原生 2.三种注入 ...
- 我的NopCommerce之旅(7): 依赖注入(IOC/DI)
一.基础介绍 依赖注入,Dependency Injection,权威解释及说明请自己查阅资料. 这里简单说一下常见使用:在mvc的controller的构造方法中定义参数,如ICountryServ ...
- 大话DI依赖注入+IOC控制反转(二) 之 浅析.Net Core中的DI与IOC
转发时请注明原创作者及地址,否则追究责任.原创:alunchen 在上一篇文章中,我们聊了很多关于定义的方面,比较孤燥,下面我们结合.Net Core聊一下依赖注入&控制反转. 三种对象生 ...
- DI依赖注入/IOC控制反转
DI依赖注入# 啥都不说,直接上代码 <?php class UserController { private $user; function __construct(UserModel $us ...
- Unity 依赖注入之二
1. 构造子注入 1.1 构造子注入初级代码 container.RegisterType<IMyWork, MyWork>(new InjectionConstructor(new Bo ...
- 关于依赖注入IOC/DI的感想
之前一直不明白依赖注入有什么好处,甚至觉得它是鸡肋,现在想想,当时真是可笑. 这个想法正如同说接口是没有用处一样. 当整个项目非常庞大,各个方法之间的调用非常复杂,那么,可以想象一下,假设说没有任何的 ...
- Spring控制反转与依赖注入(IOC、DI)
IOC: 反转控制 Inverse Of Control DI:依赖注入 Dependency Injection 目的:完成程序的解耦合 解释:在应用系统的开发过程中,有spring负责对象的创 ...
- 轻松了解Spring中的控制反转和依赖注入(二)
紧接上一篇文章<轻松了解Spring中的控制反转和依赖注入>讲解了SpringIOC和DI的基本概念,这篇文章我们模拟一下SpringIOC的工作机制,使我们更加深刻的理解其中的工作. 类 ...
随机推荐
- 它们的定义app.config中间section节点和在执行中使用
如果现在我们需要在app.config一个节点的在下面的例子中,定义,我们需要如何进行操作? <configSections> <section name="integra ...
- REUSEADDR 选项
一般而言,对于处理2MSL状态的套接字(一般为服务端套接字)是不允许接受从同一客户端重新发起一个新的连接的,但是套接字编程系统接口允许应用程序通过设置一个REUSEADDR选项,使处于2MSL状态的套 ...
- 我的MYSQL学习心得(十一)
原文:我的MYSQL学习心得(十一) 我的MYSQL学习心得(十一) 我的MYSQL学习心得(一) 我的MYSQL学习心得(二) 我的MYSQL学习心得(三) 我的MYSQL学习心得(四) 我的MYS ...
- 枚举 UIButton补充
一.URL 1.什么是URL? URL是某个资源的唯一路径,通过这个路径就能访问对应的资源 2.URL的组成 协议头://全路径 * 协议头就代表资源的类型,比如http代表网络服务器资源,ftp代表 ...
- Notepad++ 配置 支持jquery、html、css、javascript、php代码提示
原文:Notepad++ 配置 支持jquery.html.css.javascript.php代码提示 官网下载:http://notepad-plus-plus.org/ 获取插件的方法:打开软件 ...
- Critical thinking and Thoughtful writing
Critical thinkers are able to : Articulate their ideas clearly and persuasively in writing Understan ...
- 一步一步写算法(之prim算法 中)
原文:一步一步写算法(之prim算法 中) [ 声明:版权所有,欢迎转载,请勿用于商业用途. 联系信箱:feixiaoxing @163.com] C)编写最小生成树,涉及创建.挑选和添加过程 MI ...
- Android自动化测试框架新书:交流
大家觉得编写一本描述MonkeyRunner原理分析的书籍如何?估计大概10万字左右.内容大概分布如下: Monkey实现原理: 去描述运行在目标安卓机器的monkey是如何运行并处理MonkeyRu ...
- leetcode 34 Search Insert Position
Given a sorted array and a target value, return the index if the target is found. If not, return the ...
- 安装SQL Server 2005 - 初学者系列 - 学习者系列文章
初学者阶段,建议从数据库为基础入手进行学习. 下面介绍微软的SQL Server 2005数据库的安装. 首先,从下列地址获取SQL Server 2005的安装程序. ed2k://|file|cs ...