概述

        在进行WPF开发时,我遇到了Unity Container、IoC、Dependency Injection等。当时我陷入到困惑的思考总,为什么要使用这些。当后来我逐渐的了解这些技术的优点后,我开始意识到了我们实际上是需要他们的。

        本文,我来解释DI和IoC的需求和使用情况,本文分为5个部分。

        1. 依赖倒置原则

        2. 控制反转和控制反转容器(IoC Container)

        3. 自定义控制反转容器(IoC Container)

        4. 自定义支持生命周期的控制反转容器(IoC Container)

        5. 利用Microsoft Unity的依赖注入(Dependency Injection)

前提条件

        最好是最如下内容有所了解

        开闭原则(Open/closed principle)

        接口分离原则

依赖倒置原则(DIP)

        DIP是SOLID原则中的一个,他们由Robert Martin于1992年提出。

        S – Single responsibility principle(单一职责原则)

        O – Open/closed principle(开闭原则)

        L – Liskov  substitution principle(里氏代换原则)

        I – Interface segregation principle(接口分离原则)

        D – Dependency inversion principle(依赖倒置原则)

C. Robert Matrin的依赖倒置原则内容如下:

        a). 高层次的模块不应该依赖低层次的模块,他们都应该依赖于抽象。

        b). 抽象不应该依赖于具体,具体应该依赖于抽象。

DIP指的就是要依赖于抽象,不要依赖于具体。简单的说就是要求对抽象进行编程,不要对实现进行编程,这样就降低了客户与实现模块间的耦合。

iOS 组件反向依赖_控制反转

图 1

如图1所示,“Copy”程序(高层次的模块)会调用“Read Keyboard”和“Write Printer”。因此“Copy”依赖“Read Keyboard”和“Write Printer”,并且存在紧耦合。

public class Copy
    {
        publicvoidDoWork()
        {
            ReadKeyboardreader =new ReadKeyboard();
            WritePrinterwriter =new WritePrinter();
 
            string data= reader.ReadFromKeyboard();
            writer.WriteToPrinter(data);
        }
    }

        上述代码看起来还不错,但是当我们需要追加更多的Reader或者Writer对象时,在上述代码结构下,我们需要修改“copy”程序,增加创建新Reader或Writer对象,并且要追加条件判断语句,来选择用哪个Reader或Writer来执行操作。但是这么做的话,违背了面向对象的开闭原则。

        例如,我们扩展“Copy”程序,如图2,此时我们的“copy”程序可以从扫描器读数据并能够将数据写入到磁盘。

iOS 组件反向依赖_设计模式_02

图 2

        在这种情况下,我们需要调整我们的“copy”程序

public class Copy
    {
        publicvoidDoWork()
        {
            string data=string.Empty;
            switch(readerType)
            {
                case"keyboard":
                    ReadKeyboardreader = newReadKeyboard();
                    data =reader.ReadFromKeyboard();
                    break;
                case"scanner":
                    ReadScannerreader2 = newReadScanner();
                    data =reader2.ReadFromScanner();
                    break;
            }
            switch(writerType)
            {
                case"printer":
                    WritePrinterwriter = newWritePrinter();
                   writer.WriteToPrinter(data);
                    break;
                case"flashdisk":
                    WriteFlashDiskwriter2 = newWriteFlashDisk();
                    writer2.WriteToFlashDisk(data);
                    break;
            }
        }
 
        privatestringreaderType;
        privatestringwriterType;
    }

        根据上面的场景,如果需要继续追加新的Reader和Writer,我们就需要修正copy程序,因此copy严重的依赖具体的Reader和Writer对象。

        为了解决上述问题,我们可以调整我们的“copy”程序,使其依赖抽象的接口而不是具体的实现。下图解释了依赖倒置。

iOS 组件反向依赖_C#_03

图 3

iOS 组件反向依赖_iOS 组件反向依赖_04

图 4

在图3、4中,“Copy”程序依赖两个抽象的接口IReader和IWriter,这样低级别的组件就被抽象成了接口。

如下例子,在上述的图中,可以看出,类“ReadKeyboard”派生自接口IReader,类“WritePrinter”派生自接口IWriter,因此“Copy”程序调用IReader和IWriter就执行具体的操作了。因此如果我们想要追加更多的底层组件,例如“Scanner”和“FlashDisk”,我们可以创建派生自“IReader”和“IWriter”的新类即可。

public interface IReader
    {
        stringRead();
    }
 
    public interface IWriter
    {
        voidWrite(string data);
    }
 
    public class ReadKeyboard :IReader
    {
        publicstringRead()
        {
            // code to read from keyboardand return as string
            returnstring.Empty;
        }
    }
 
    public class ReadScanner :IReader
    {
        publicstringRead()
        {
            // code to read from scannerand return as string
            returnstring.Empty;
        }
    }
 
    public class WritePrinter :IWriter
    {
        publicvoidWrite(string data)
        {
            // code to write to theprinter
        }
    }
 
    public class WriteFlashDisk :IWriter
    {
        publicvoidWrite(string data)
        {
            // code to write to the flashdisk
        }
    }
 
    public class Copy
    {
        privatestring_readerType;
        privatestring_writerType;
 
        public Copy(stringreaderType,string writerType)
        {
            _readerType = readerType;
            _writerType = writerType;
        }
 
        publicvoidDoWork()
        {
            IReaderreader =null;
            IWriterwriter =null;
            string data;
            switch(readerType)
            {
                case"keyboard":
                    reader = new ReadKeyboard();
                    break;
                case"scanner":
                    reader = new ReadScanner();
                    break;
            }
 
            switch(writerType)
            {
                case"printer":
                    writer = new WritePrinter();
                    break;
                case"flashdisk":
                    writer = new WriteFlashDisk();
                    break;
            }
 
            data = reader.Read();
            writer.Write(data);
        }
    }

        在这种情况下,虽然具体的读写操作已经被抽象出来的,但是仍然存在高层模块依赖底层模块的现象,因为我们在高层模块中明确的实例化了具体的底层类对象。因此这种情况下,当追加新的类时,代码中的swith、case处仍然需要不断的维护。因此上述的修改方案并未完全的满足DIP原则。

        为了彻底的移除依赖,我们需要在高层组件外面来创建依赖对象,并且依赖对象中存在某种机制能够传递到依赖模块中。

        现在新的问题来了,如何实现依赖反转(Dependency Inversion)

        其中一个方法是控制反转(Inversion ofControl, IoC),相当于如下代码:

public class Copy
    {       
        publicvoidDoWork()
        {
            IReaderreader = serviceLocator.GetReader();
            IWriterwriter = serviceLocator.GetWriter();
            string data= reader.Read();
            writer.Write(data);
        }
    }

        上述代码替代了实例化Reader和Writer对象的代码,上述代码已经将创建Reader和Writer对象的处理转移到servicelocator中。从而使得“copy”程序不必因为底层组件的增减而修改了。

        依赖注入(DependencyInjection)是实现控制反转的一种机制,在下一部分我会介绍什么是控制反转(Inversionof Control, IoC)及利用不同的机制实现依赖倒置原则(其中依赖注入是其中的一种办法)。

小结

        本文介绍了依赖倒置原则,及其在真实的使用场景。在后面的文章中我会介绍控制反转(Inversionof Control,IoC)和依赖注入(DependencyInjection,DI)。