C#/.net 单元测试xUnit、Mock、Moq

在做单元测试的时候,有时需要引用很多的外部对象,例如网络通信、记录日志等。单元测试无法控制这些外部的依赖对象,所以需要使用Stub和Mock来模拟这些外部对象。

案例说明

用以下实例来进行单元测试

//检查文件名的长度,并使用IWebService来记录错误
public class CheckLength
{
public IWebService webService { get; set; }
public IEmailService emailService { get; set; }

public void Analyze(string fileName)
{
if (fileName.Length<8)
{
webService.LogError("文件名称太短");
}
else
{
emailService.SendEmail("123@qq.com", "文件名符合要求");
}
}
}

我们发现webService和emailService都是外部依赖对象,需要我们自己写Stub和Mock来模拟这两个外部对象

public class StubWebService : IWebService
{
public void LogError(string message)
{
Debug.WriteLine(message);
}
}

public class MockEmailService : IEmailService
{
public string Email { get; set; }
public string message { get; set; }
public void SendEmail(string email, string message)
{
this.Email= email;
this.message = message;
}
}

然后进行单元测试

[Fact()]
public void AnalyzeTest()
{
//arange
StubWebService stubService = new StubWebService();
MockEmailService mockService = new MockEmailService();

CheckLength checkLength = new CheckLength();
//将自定义的服务赋给checkLength
checkLength.emailService = mockService;
checkLength.webService= stubService;


//act
string fileName = "a3t";
checkLength.Analyze(fileName);

//assert
Assert.Equal("123@qq.com", mockService.Email);
Assert.Equal("文件名符合要求", mockService.message);
}

stub和Mock的区别

从上面可以看出,Stub是完全模拟一个外部依赖(直接输出,无法进行断言),而Mock则是用来断言

注意事项:

要以接口为依赖对象,例如本案例中

public IWebService webService { get; set; }
public IEmailService emailService { get; set; }

Moq是.net平台下的一个非常流行的模拟库,目前依赖注入模式非常流行,有时被测试的类或者方法需要注入数十项接口,如果像上面那样将接口重写为自定义的“假实现”,也要写大量的代码,而利用moq框架可以自动生成接口的代理对象,减少工作量。

​moq框架教程​