TDD和模拟TcpClient
时间:2020-03-06 14:53:34 来源:igfitidea点击:
人们如何模仿TcpClient(或者类似TcpClient之类的东西)?
我有接受TcpClient的服务。我应该将其包装在其他更可笑的东西中吗?我应该如何处理?
解决方案
当进入不适合测试的模拟类时(即密封/未实现任何接口/方法不是虚拟的),我们可能希望使用适配器设计模式。
在此模式中,我们将添加一个实现接口的包装类。然后,我们应该模拟该接口,并确保所有代码都使用该接口而不是不友好的具体类。它看起来像这样:
public interface ITcpClient
{
Stream GetStream();
// Anything you need here
}
public class TcpClientAdapter: ITcpClient
{
private TcpClient wrappedClient;
public TcpClientAdapter(TcpClient client)
{
wrappedClient = client;
}
public Stream GetStream()
{
return wrappedClient.GetStream();
}
}
使用适配器模式绝对是解决问题的标准TDD方法。但是,我们也可以只创建TCP连接的另一端,并让测试工具驱动该连接。
IMO适配器类的广泛使用使设计中最重要的部分变得模糊不清,并且还倾向于从测试中删除很多本应在上下文中进行测试的内容。因此,替代方法是建立测试支架,以包括更多正在测试的系统。如果我们是从头开始构建测试,那么我们仍将能够将失败原因隔离到给定的类或者函数中,而不会孤立...
我认为@Hitchhiker走在正确的道路上,但我也想考虑将类似的东西抽象出来只是一步之遥。
我不会直接模拟TcpClient,因为即使我们已经编写了测试,这仍然会使我们与基础实现过于紧密地联系在一起。也就是说,实现专门与TcpClient方法绑定。就个人而言,我会尝试这样的事情:
[Test]
public void TestInput(){
NetworkInputSource mockInput = mocks.CreateMock<NetworkInputSource>();
Consumer c = new Consumer(mockInput);
c.ReadAll();
// c.Read();
// c.ReadLine();
}
public class TcpClientAdapter : NetworkInputSource
{
private TcpClient _client;
public string ReadAll()
{
return new StreamReader(_tcpClient.GetStream()).ReadToEnd();
}
public string Read() { ... }
public string ReadLine() { ... }
}
public interface NetworkInputSource
{
public string ReadAll();
public string Read();
public string ReadLine();
}
此实现将使我们与Tcp相关的细节完全脱钩(如果这是设计目标),甚至可以从硬编码的一组值或者测试输入文件中引入测试输入。如果我们正准备长期测试代码,请非常有帮助。

