Dependency Injection
Inversion of Control
- Dependency Injection
- Dependency Lookup
loose coupling/maintainability/
late binding
abstract factory
unit test
container
非 DI 的版本
底下這段程式碼是 Console 應用程式的進入點:
class Program
{
static void Main(string[] args)
{
HeavyDuty aTask = new HeavyDuty();
aTask.Run();
}
}
Main 函式會先建立類別 HeavyDuty 的執行個體,然後呼叫該物件的 Run 方法。HeavyDuty 類別的原始碼如下:
public class HeavyDuty
{
private ConsoleLogger logger; public HeavyDuty()
{
// 在建構式裡面就先建立好欲使用的記錄器.
logger = new ConsoleLogger();
} public void Run()
{
logger.WriteEntry("HeavyDuty is running...");
}
}
從 HeavyDuty 類別的原始碼可以發現,它使用了另一個叫做 ConsoleLogger 的類別來當作記錄器,以便輸出一些訊息。ConsoleLogger 的責任很簡單,就只是將指定的訊息輸出至 console 視窗而已:
public class ConsoleLogger
{
public void WriteEntry(string msg)
{
Console.WriteLine(msg);
}
}
整理一下:主程式會用到 HeavyDuty 類別來執行某項工作,而 HeavyDuty 又會使用 ConsoleLogger 來輸出 log 訊息。三者關係如下:

這個例子的情境是:應用程式常常會需要寫 log,而寫 log 的機制有好多種,例如本例的 ConsoleLogger 是將 log 訊息輸出至 console 視窗,其他可能的 log 方式還有:寫入 Windows 事件日誌、發送 e-mail、寫入資料庫等等。
問題來了,此例的 ConsoleLogger 是由 HeavyDuty 類別所建立,並非由主程式控制,如果應用程式中還有其他類別需要寫 log,也就必須像 HeaveyDuty 那樣,在類別裡面建立 ConsoleLogger 的物件實體並呼叫其方法。如此一來,若有 N 個類別要寫 log,就有 N 個類別相依於 ConsoleLogger 類別。萬一有一天要改成寫入 Windows 事件日誌(可能會設計另一個 WindowsEventLogger 類別),這要改多少程式碼呀?
接著就來看看 DI 如何處理這個問題。
改成 DI 版本
看過了非 DI 版本的程式寫法以及類別圖,我們知道未來可能會有很多類別會相依於 ConsoleLogger 這個具象類別(concrete class),而這層相依性,極可能造成日後很高的維護成本。因此,我們的首要目標就是減輕、甚至消除這層相依性,或者說:解耦合(decouple)。
前兩篇曾提過,介面是解耦合的一種很好用的工具。故我們可以先把「寫入 log」這個操作放到一個介面中,讓所有要寫 log 的類別只針對一個標準介面來操作。就將此介面命名為 ILogger 好了。程式碼很簡單,就只有一個方法:
public interface ILogger
{
void WriteEntry(string msg);
}
然後,原本的 ConsoleLogger 類別(以及其他要提供寫 log 操作的類別)必須實作此介面:
public class ConsoleLogger : ILogger
{
public void WriteEntry(string msg)
{
Console.WriteLine(msg);
}
}
接下來,我們希望 HeavyDuty 類別(以及未來其他需要寫 log 的類別)只依賴 ILogger 介面,而不要依賴特定實作。因此,原先在 HeavyDuty 中建立 ConsoleLogger 物件實體的寫法就必須拿掉。可是,要使用物件之前,一定得在某個地方先建立好物件的實體才行啊。那麼,要在哪裡、由誰來建立物件呢?
建立 logger 物件的工作,由於需要用到具象類別,此動作會產生相依性。我們希望將依賴程度盡量降低,因此我們可以將此相依性從 HeavyDuty 類別中抽離,轉移至主程式。換言之,由主程式來建立真正的 logger 物件實體。那麼,HeavyDuty 只要有一個指向實際 logger 物件的參考就夠了--這很簡單,只要主程式在建立 HeavyDuty 物件時,透過建構式的參數傳入 logger 物件參考就解決了。
所以修改後的 HeavyDuty 類別會像這樣:
public class HeavyDuty
{
private ILogger logger; public HeavyDuty(ILogger aLogger)
{
logger = aLogger;
} public void Run()
{
logger.WriteEntry("HeavyDuty is running...");
}
}
有注意到嗎?類別裡面完全沒有指涉任何 logger 類別,而只用到 ILogger 介面而已。真正的物件實體,是透過建構式的參數傳進來。這種透過建構式來提供(注入)相依物件的作法,叫做「建構式注入」(Constructor Injection)。
最後是主程式的 Main 方法:
class Program
{
static void Main(string[] args)
{
ILogger logger = new ConsoleLogger();
HeavyDuty aTask = new HeavyDuty(logger);
aTask.Run();
}
}
到這裡應該完全清楚了。若還覺得有點模糊,可試著倒著順序往回逐一檢視程式碼,應該也能理出一些頭緒。了解各類別之間的關係之後,也就不難整理出底下的類別圖了。

你也可以從這張類別圖,搭配程式碼來推敲各類別的關聯。同時想想看為甚麼要這樣設計,這樣設計有什麼好處(前面都有提到)。
Dependency Injection的更多相关文章
- Ninject学习(一) - Dependency Injection By Hand
大体上是把官网上的翻译下而已. http://www.ninject.90iogjkdcrorg/wiki.html Dependency Injection By Hand So what's Ni ...
- MVC Controller Dependency Injection for Beginners【翻译】
在codeproject看到一篇文章,群里的一个朋友要帮忙我翻译一下顺便贴出来,这篇文章适合新手,也算是对MEF的一个简单用法的介绍. Introduction In a simple stateme ...
- 控制反转Inversion of Control (IoC) 与 依赖注入Dependency Injection (DI)
控制反转和依赖注入 控制反转和依赖注入是两个密不可分的方法用来分离你应用程序中的依赖性.控制反转Inversion of Control (IoC) 意味着一个对象不会新创建一个对象并依赖着它来完成工 ...
- [转载][翻译] IoC 容器和 Dependency Injection 模式
原文地址:Inversion of Control Containers and the Dependency Injection pattern 中文翻译版本是网上的PDF文档,发布在这里仅为方便查 ...
- Inversion of Control Containers and the Dependency Injection pattern(转)
In the Java community there's been a rush of lightweight containers that help to assemble components ...
- Scala 深入浅出实战经典 第57讲:Scala中Dependency Injection实战详解
王家林亲授<DT大数据梦工厂>大数据实战视频 Scala 深入浅出实战经典(1-87讲)完整视频.PPT.代码下载:百度云盘:http://pan.baidu.com/s/1c0noOt6 ...
- 【译】Dependency Injection with Autofac
先说下为什么翻译这篇文章,既定的方向是架构,然后为了学习架构就去学习一些架构模式.设计思想. 突然有一天发现依赖注入这种技能.为了使得架构可测试.易维护.可扩展,需要架构设计为松耦合类型,简单的说也就 ...
- 依赖注入 | Dependency Injection
原文链接: Angular Dependency Injection翻译人员: 铁锚翻译时间: 2014年02月10日说明: 译者认为,本文中所有名词性的"依赖" 都可以理解为 & ...
- 黄聪:Microsoft Enterprise Library 5.0 系列教程(八) Unity Dependency Injection and Interception
原文:黄聪:Microsoft Enterprise Library 5.0 系列教程(八) Unity Dependency Injection and Interception 依赖注入容器Uni ...
随机推荐
- phpcms V9实现wap上一篇、下一篇功能
在phpcms\modules\wap\index.php里面,搜索上面这句 if(!$r || $r['status'] != 99) showmessage(L('info_does_not_ex ...
- ASP.NET 5探险(8):利用中间件、TagHelper来在MVC 6中实现Captcha
(此文章同时发表在本人微信公众号"dotNET每日精华文章",欢迎右边二维码来关注.) 题记:由于ASP.NET 5及MVC 6是一个微软全新重新的Web开发平台,之前一些现有的验 ...
- .deb文件打包
最近因项目需要,需要把文件夹打包为.deb格式的包,幸亏一位朋友帮忙指导了我一个晚上,才得以完成,这里再次对他表示感谢. 整理打包流程如下: 请先参考此博客内容,了解deb文件打包 如何制作Deb包和 ...
- ARM伪指令,王明学learn
ARM伪指令 在ARM汇编语言程序中里,有一些特殊指令助记符与指令系统的助记符不同,没有相对应的操作码,通常称这些特殊指令助记符为伪指令,他们所完成的操作称为伪操作.伪指令在元程序中的作用是为完成汇编 ...
- C++ find 函数用法
头文件 #include <algorithm> 函数实现 template<class InputIterator, class T> InputIterator find ...
- 湖南省第十二届大学生计算机程序设计竞赛 B 有向无环图 拓扑DP
1804: 有向无环图 Time Limit: 5 Sec Memory Limit: 128 MBSubmit: 187 Solved: 80[Submit][Status][Web Board ...
- felx项目属性(二)
order flex-grow flex-shrink flex-basis flex align-self 1.1 order css order属性规定了弹性容器中的可伸缩项目在布局时的顺序.元素 ...
- 【MySQL 忘记密码】MySQL忘记密码怎么解决 mysql5.5 windows7
---恢复内容开始--- 如果MySQL 长久不使用,忘记密码,怎么解决??? 1.首先,需要在任务管理器关闭mysql相关的服务进程 2.cmd,进入DOS窗口,进入到mysql的安装路径的bin目 ...
- java net编程
转自:http://www.cnblogs.com/linzheng/archive/2011/01/23/1942328.html 一,网络编程中两个主要的问题 一个是如何准确的定位网络上一台或多台 ...
- 6.android加密解析
编码.数字摘要.加密.解密 UrlEncoder /Urldecoder String str = "http://www.baidu.com?serach = 哈哈"; Stri ...