(1)构造注入(Constructor Injection)
在我们创建Manager对象的时候,将记录日志的对象作为构造参数传递给新创建的Manager对象,假设Manager有一个带ILog类型参数的构造方法,如:
class Manager { ILog _logger; public Manager(ILog logger) { _logger = logger; } //… }
那么,我们在创建Manager对象的时候,这样编写代码:
Manager m = new Manager(new FileLogger());
//Manager m = new Manager(new EmailLogger());
//Manager m = new Manager(new NotifyLogger());
很明显,这种日志记录方式一直不变,对Manager终生有效。
(2)方法注入(Method Injection)
为Manager类中每个需要记录日志的方法增加一个ILog的参数,比如Manager.DoSomething方法重新定义为:
class Manager { //… public void DoSomething(ILog logger) { try { //… } catch(Exception ex) { logger.Log(ex.ToString()); } } }
那么我们之后在使用Manager的时候,每次调用方法都应该为它提供一个记录日志的对象,如:
Manager m = new Manager();
m.DoSomething(new FileLogger());
m.DoSomething(new EmailLogger());
m.DoSomething(new NotifyLogger());
这种记录日志的方式,只对当前方法有效,每次调用方法都可以不同。
(3)属性注入(Property Injection)
在Manager类中公开一个属性,用来设置日志记录对象,Mananger这样定义:
class Manager { private ILog _logger; public ILog Logger { get { return _logger; } set { _logger = value; } } //… }
之后我们使用Mananger时,可以随时更换它的日志记录方式:
Mananger m = new Manager();
m.Logger = new FileLogger();
m.Logger = new EmailLogger();
m.Logger = new NotifyLogger();
使用这种方式,我们可以随时切换记录日志的方式,它的灵活度介于“构造注入”和“方法注入”之间。
以上三种依赖注入方法可以混合使用,也就是说,你可以为Manager类定义一个带ILog类型的参数,同时也可以定义一个ILog类型的属性,或者为每个方法增加一个ILog类型的参数。
注:
【1】在.NET中,“抽象层”可以不使用接口interface去实现,而是直接使用委托,举一个例子,我们使用FileStream.BeginRead方法时,给它提供的一个AsyncCallback回调参数,其实就是属于“方法注入”的一种。
【2】类型与类型之间不可能完全失去依赖关系,怎样让这种非有不可的依赖关系更微弱,是软件设计的一门高深学问。