使用Unity解耦你的系统—PART3——依赖注入
继续学习Unity,在前几篇中已经将Unity的使用方法做了一个还算详细的介绍了,主要是如何通过代码及配置文件来注册对象之间的关系、Unity内置所有的生命周期管理使用介绍,及Unity的Register和Resolve的一些高级应用。通过在PART1——为什么要使用Unity?的学习我们知道Unity可以帮我们简化并管理对象之间的关系(也就是前几篇所介绍的),而今天则要介绍Unity的另外一个重要功能——DI(依赖注入)。
本篇文章将主要介绍:
1、构造函数注入。
2、属性注入。
3、方法注入。
4、使用配置完成各种注入。
5、对已经创建对象进行注入。
一、构造函数注入
在有些时候,我们所编写的类中的构造函数都会包含对其他对象的引用,如下代码:
public Subject2(MyClass myClass1,MyClass myClass2)
{
myClass1.Name = "班级1";
myClass2.Name = "班级2";
}
可以看到这个构造函数有2个参数,都依赖于MyClass类,如果一般情况下想要调用,我们总是需要实现构建好2个myclass对象,所以相对来说比较麻烦,而如果使用Unity来调用就会方便许多,如下代码:
container.Resolve<Subject2>();
仅仅一行既可,Unity容器会自动帮我们构建好所需的依赖对象实例。
当然这个只是简单的使用,在实际的情况下我们不会这么编写代码,我们不会直接引用对象,而是直接引用接口,这样可以解除代码的耦合性,如下代码:
public Subject2(IClass myClass1)
{
myClass1.Name = "班级1";
}
这样Subject2类的构造函数仅仅依赖于IClass接口,并依赖于具体的实现类,这种情况下,如果想调用的话,需要实现注册好对象之间的关系,如下代码:
public static void ConstructorInjection()
{
//container.Resolve<Subject2>(); container.RegisterType<IClass, MyClass>();
container.Resolve<Subject2>();
}
通过Unity容器对象的获取对象时候默认总是获取非命名默认注册对象,但是如果调用Subject2的构造函数参数我想指定具体的对象时怎么办?
这时就需要使用Dependency这个特性类来解决了,在需要特殊指定的依赖关系的参数上加上Dependency特性,并为Dependency指定好参数name(此name参数表示注册对象关系时所指定的名称),代码如下:
public Subject2([Dependency("your")]
IClass classInfo)
{
classInfo.Name = "班级1";
}
调用代码:
container.RegisterType<IClass, MyClass>()
.RegisterType<IClass, YourClass>("your");
container.Resolve<Subject2>();
可以看到,我对IClass注册了2个对象,而我在Subject2的构造函数参数上使用了Dependency特性指定了ClassInfo参数获取注册名为“your”的对象。
现在还有一个问题,就是当一个类存在多个构造函数的时候,我们如何区分哪个构造函数需要实现注入,哪个不需要?
这时就可以使用InjectionConstructor特性来标识,代码如下:
[InjectionConstructor]
public Subject2([Dependency("your")]
IClass classInfo)
{
classInfo.Name = "班级1";
} public Subject2(IClass classInfo1, IClass classInfo2)
{
classInfo1.Name = "班级1";
classInfo2.Name = "班级2";
}
Unity只会调用标识了InjectionConstructor特性的构造函数,这样就很好的解决了多构造函数的情况下,Unity调用哪个构造函数。
二、属性注入
属性注入和构造函数注入类似,只需在需要注入的属性上增加一个Dependency特性,同样的也可以为Dependency指定一个name参数用来指 定注入属性的具体对象,如下代码,在Subject属性上增加了Dependency特性,来表示这个属性需要注入:
public class MyClass : IClass
{
public MyClass()
{
} public void ShowInfo()
{
Console.WriteLine("这个是我的班级");
} [Dependency("Subject1")]
public ISubject Subject { get; set; } public string Name { get; set; } public string Description { get; set; }
}
具体的调用代码:
public static void PropertyInjection()
{
container.RegisterType<ISubject, Subject1>("Subject1")
.RegisterType<IClass,MyClass>();
var classInfo = container.Resolve<IClass>();
Console.WriteLine(classInfo.Subject.Name);
}
这样ClassInfo的Subject属性自动关联到了Subject1类上(完成了属性注入),访问classInfo.Subject.Name可以得到“科目1”。
三、方法注入
方法注入同样只需在需要注入的方法上增加一个特性——InjectionMethod既可(其使用方法也和构造注入类似),这样Unity会自动帮我们完成注入,方法注入和构造注入一样,同样可以在方法的参数上指定Dependency特性来指定参数所依赖的注册,下面的类代码中包含了构造注入、属性注入及方法注入,这边集合在一起展示相对来说直观一些:
public class MyClass : IClass
{
public MyClass()
{
} public MyClass(ISubject subjectInfo)
{
this.TempSubject1 = subjectInfo;
} public void ShowInfo()
{
Console.WriteLine("构造注入成功后临时科目1名称:" + this.TempSubject1.Name);
Console.WriteLine("属性注入成功后临时科目名称:" + this.Subject.Name);
Console.WriteLine("方法注入成功后临时科目2名称:" + this.TempSubject2.Name);
} [InjectionMethod]
public void Init(ISubject subjectInfo)
{
TempSubject2 = subjectInfo;
} [Dependency("Subject1")]
public ISubject Subject { get; set; } public ISubject TempSubject1 { get; set; }
public ISubject TempSubject2 { get; set; } public string Name { get; set; } public string Description { get; set; }
}
具体的调用代码:
public static void MethodInjection()
{
container.RegisterType<ISubject, Subject3>("Subject1")
.RegisterType<ISubject, Subject4>();
container.RegisterType<IClass, MyClass>(); var classInfo = container.Resolve<IClass>();
classInfo.ShowInfo();
}
效果图如下:
4、使用配置完成各种注入
上面所演示的代码都是通过代码来完成对象的注入,下面演示下如何通过配置文件来配置这些注入,具体配置代码如下:
<!--依赖注入配置,包括构造注入,方法注入和属性注入-->
<alias alias="IClass" type="UnityStudyConsole.IDemo.IClass, UnityStudyConsole" />
<alias alias="MyClass" type="UnityStudyConsole.Demo.MyClass, UnityStudyConsole" />
<alias alias="ISubject" type="UnityStudyConsole.IDemo.ISubject, UnityStudyConsole" />
<alias alias="Subject3" type="UnityStudyConsole.Demo.Subject3, UnityStudyConsole" />
<alias alias="Subject4" type="UnityStudyConsole.Demo.Subject4, UnityStudyConsole" />
<container name="Third">
<register type="IClass" mapTo="MyClass">
<constructor>
<param name="subjectInfo" type="ISubject">
<dependency name="subjectInfo" type="Subject4"/>
</param>
</constructor>
<method name="Init">
<param name="subjectInfo" type="ISubject">
<dependency name="subjectInfo" type="Subject4"/>
</param>
</method>
<property name="Subject">
<dependency name="Subject1" type="Subject3"/>
</property>
</register>
</container>
代码如下:
public static void DIConfiguration()
{
//获取特定配置节下已命名的配置节<container name="Third">下的配置信息
container.LoadConfiguration("Third");
var classInfo = container.Resolve<IClass>();
classInfo.ShowInfo();
}
具体的效果是和上面截图中是一样的,只不过这边的依赖注入是通过配置文件来实现的。
5、对已经创建对象进行注入
一般来说如果想实现依赖注入需要通过Unity容器来进行对象注册,然后通过Unity容器来获取对象,但是如果对象已经存在(就是不是通过Unity容器来获取的对象),这时如何来通过Unity来实现对已有的对象进行依赖注入呢?
Unity容器已经为我们提供好了这种情况的解决办法,就是BuildUp方法,看下下面的代码就能明白了:
public static void BuildUp()
{
//事先注册好ISubject和MySubject之间的关系
//并指定一个名称以方便在接口中的属性上应用[Dependency("Subject1")]特性
//具体请见IDemo.IClass
container.RegisterType<ISubject, Subject1>("Subject1");
IClass classInfo = new MyClass();
IClass classInfo2 = container.BuildUp<IClass>(classInfo);
Console.WriteLine(classInfo2.Subject.Name);
}
#endregion
在上面的代码中,已经创建好一个对象实例了,这时只需将这个对象作为参数放入BuildUp中,同时还需指定这个对象实例所实现的接口或父类类型,这样Unity就会自动帮我们将这个已存在对象实例中各种注入。
需要注意的是,与上面所说3种依赖注入的不同的是,上面3中的依赖注入需要在具体的类中使用InjectionConstructor、 InjectionMethodDependency和特性,而如果对已经存在的对象进行依赖注入,则需要将InjectionConstructor、 InjectionMethod和Dependency写在这个对象实例所实现的接口或父类中,否则会报错!
到本文为止,Unity的各种常用功能都已经介绍完毕了,各位可以下载下面的源代码进行查看,同时也可以通过Main方法中的各个方法来查看Unity的各种功能的使用:
示例代码下载:点我下载
使用Unity解耦你的系统—PART3——依赖注入的更多相关文章
- 使用Unity解耦你的系统—PART4——Unity&PIAB
在前面几篇有关Unity学习的文章中,我对Unity的一些常用功能进行介绍,包括:Unity的基本知识.管理对象之间的关系.生命周期.依赖注入等,今天则是要介绍Unity的另外一个重要功能——拦截(I ...
- Unity文档阅读 第二章 依赖注入
Introduction 介绍Chapter 1 outlines how you can address some of the most common requirements in enterp ...
- 【Unity】微软的一款依赖注入组件
前言 前面学习了autofac这个依赖注入组件,本来是打算写在一起的,因为这个组件没打算像autofac一样详细的写,只是写下以前自己鼓捣玩搭建框架然后使用的一个依赖注入组件,并且也是进行了封装使用. ...
- 微软企业库5.0 学习之路——第十步、使用Unity解耦你的系统—PART2——了解Unity的使用方法(3)
今天继续介绍Unity,在上一篇的文章中,我介绍了使用UnityContainer来注册对象之间的关系.注册已存在的对象之间的关系,同时着重介绍 了Unity内置的各种生命周期管理器的使用方法,今天则 ...
- 使用Unity 实现ASP.NET Web API 依赖注入
DI/IoC 的设计前面已经讲过好几次了,简单的一段话说明就是:「目标对象与外部相依的方式仅相依于 interface,而相依 interface 的 instance 透过 constructor ...
- [.NET] 使用Json.NET提供依赖注入功能(Dependence Injection)
[.NET] 使用Json.NET提供依赖注入功能(Dependence Injection) 前言 在一些小型项目的开发情景里,系统不需要大型DI Framework所提供的:单一对象生成.生命周期 ...
- IOC使用Unity 实现依赖注入
转自:http://www.cnblogs.com/techborther/archive/2012/01/06/2313498.html http://www.cnblogs.com/xishuai ...
- 理解依赖注入(IOC)和学习Unity
资料1: IOC:英文全称:Inversion of Control,中文名称:控制反转,它还有个名字叫依赖注入(Dependency Injection). 作用:将各层的对象以松耦合的方式组织在一 ...
- 依赖注入与Unity(一) 介绍
在你学习依赖注入和Unity之前,你需要明白你为什么要使用它们.为了明白为什么要使用它们,你应该明白依赖注入和Unity能够帮助你解决什么类型的问题.作为介绍部分,这一章不会涉及太多关于Uni ...
随机推荐
- python学习(十三)进程和线程
python多进程 from multiprocessing import Process import os def processFunc(name): print("child pro ...
- 对于redis框架的理解(三)
上一篇讲完了initServer的大体流程,其中aeCreateEventLoop(),这个函数 没有详细说明,我们在这一篇里讲述Ae.h和Ae.c, 这里面的api阐述了如何创建 eventLoop ...
- centos pure-ftpd配置及错误解决
使用yum安装pure-ftpd Pure-FTPd是Linux上的一个开源的FTP服务程序,在易用性.配置性上比vsftp较方便,下面我们使用centos6演示安装和配置pure-ftpd. 安装e ...
- 条件转化,2-sat BZOJ 1997
http://www.lydsy.com/JudgeOnline/problem.php?id=1997 1997: [Hnoi2010]Planar Time Limit: 10 Sec Memo ...
- Ping命令网络监测
按照由近到远原则: 1. ping 127.0.0.1 先检查TCP/IP协议栈是否正常. 2. ping 本地ip 检查网卡是否工作正常. 3. ping 网关地址 检查和网关连接性. 4. pin ...
- AlloyTouch 简介
AlloyTouch 是来自于腾讯AlloyTeam团队开发的一个适用用移动端的js组件库. 特性: 1.丰富的组件 选择组件.级联选择组件.轮播组件.全屏滚动组件.下拉刷新组件.上拉刷新任君选择 2 ...
- 用Matlab实现字符串分割(split)
用Matlab实现字符串分割(split)Posted on 2011/08/08 Matlab的字符串处理没有C#强大,本身又没有提供OO特性,需要依赖别的手段完成这项任务. 我们在这里借助正则表达 ...
- 【BZOJ】4766: 文艺计算姬
[题目]给定两边节点数为n和m的完全二分图,求生成树数取模给定的p.n,m,p<=10^18. [算法]生成树计数(矩阵树定理) [题解]参考自 [bzoj4766]文艺计算姬 by WerKe ...
- 【CodeForces】679 B. Bear and Tower of Cubes
[题目]B. Bear and Tower of Cubes [题意]有若干积木体积为1^3,2^3,...k^3,对于一个总体积X要求每次贪心地取<=X的最大积木拼上去(每个只能取一次)最后总 ...
- Python ctypes中cast/py_object用法
class ctypes.py_object Represents the C PyObject * datatype. Calling this without an argument create ...