抽取接口使底层实现可替换
首先将第一章的代码重构,抽取出实现代码,放到FileExtensionManager类中。
public class LogAnalyzer
{
public bool WasLastFileNameValid { get; set; }
public bool IsValidLogFileName(string fileName)
{
//使用抽取出来的类
FileExtensionManager mgr =
new FileExtensionManager();
return mgr.IsValid(fileName);
}
}
//首先定义出这个抽取的类
class FileExtensionManager
{
public bool IsValid(string fileName)
{
//读取文件
}
}
然后定义ExtensionManager的接口IExtensionManager并实现
class FileExtensionManager:IExtensionManager
{
public bool IsValid(string fileName)
{
//读取文件
}
}
//然后定义新接口
public interface IExtensionManager
{
bool IsValid(string fileName);
}
public bool IsValidLogFileName(string fileName)
{
//定义这个接口的类型变量
IExtensionManager mgr =
new FileExtensionManager();
return mgr.IsValid(fileName);
}
现在就创建了一个提供IsValid(string)方法的接口IExtensionManager,用FileExtensionManager实现这个接口。代码功能没有变,但是现在可以用自己的伪造的扩展名管理器来代替真实 扩展名管理器了
//这个是总是返回true的简单桩代码
//使用Fake说明这个类的对象类似另一个对象,既可能用作模拟对象,也可能用作桩
public class AlwaysValidFakeExtensionManager: IExtensionManager
{
//实现IExtensionManager接口
public bool IsValid(string fileName)
{
return true;
}
}
现在虽然有了一个接口和两个实现此接口的类,但是被测试的方法还是对具体类进行直接调用。
public bool IsValidLogFileName(string fileName)
{
IExtensionManager mgr =
new FileExtensionManager();
return mgr.IsValid(fileName);
}
所以必须想办法让被测试的方法调用伪对象,而不是IExtensionManager的原本实现,因此需要在代码中引入一个接缝,然后插入桩。
依赖注入:在被测试单元中注入一个伪实现
参数注入
给方法签名添加一个参数,从而给这个方法传入一个(伪造的)依赖实例
Method(string,int,FakeClass)
//其中FakeClass就是注入的依赖实例。