如题,本文主要作为在VS2012使用Fakes的入门示例,开发工具必须是VS2012或更高版本。
关于Fakes的MSDN地址:http://msdn.microsoft.com/en-us/library/hh549175.aspx
关于VS2012单元测试的前期文章:
1.《在Visual Studio 2012使用单元测试》、
2.《VS2012 单元测试之泛型类(Generics Unit Test)》、
3.《VS2012 Unit Test —— 我对接口进行单元测试使用的技巧》
4.《VS2012 Unit Test(Void, Action, Func) —— 对无返回值、使用Action或Func作为参数、多重载的方法进行单元测试》
依我个人理解单元测试就是对程序的小单元进行测试,一个测试不应包含两个或更多单元,总体而言大多都是对方法、属性的编码正确性进行验证。但是往往一个方法又会调用其他的方法或属性,我这里暂称之为外部依赖,因而外部依赖会影响程序单元的测试结果,要避免这样的情况就不得不使用一些外部依赖的模拟进行隔离(Isolate),本文就是使用了Microsoft Fakes,当然还有其他更为流行的框架可以选择使用(Moq、Rhino Mocks、Type Mock)
Fakes有两种形式:stub 和 shim。具体的介绍我就不啰嗦,因为我英文不好可能会表达错误误导新人。
我的Demo也是看了MSDN后以个人理解后进行简单的编写,如果MSDN看懂了也就不用看以下内容了,期待和我一样正在使用VS2012 MSTest进行单元测试的一起交流进步。
一、shim
以下将模拟DateTime的Now属性,假设我现在需要在活动服务类ActivityService添加一个方法验证某个线下活动是否过期。
1. 打开VS2012,创建单元测试项目FakesTesting,我这是测试先行。重命名项目自动生成的类UnitTest1为ActivityServiceTest,将TestMethod1改为IsExpireTest(是否过期).
2. 添加代码“ActivityService service = new ActivityService();”并使用VS快捷功能为我们创建ActivityService 类
3. 添加Fakes,由于DateTime位于System程序集,因而将添加System的Fake程序集(右键System程序集), 然后在测试类“using System.Fakes;”
4. 编写测试代码如下
using System;
using Microsoft.VisualStudio.TestTools.UnitTesting;
using System.Fakes;
using Microsoft.QualityTools.Testing.Fakes; namespace FakesTesting.Test
{
[TestClass]
public class ActivityServiceTest
{
[TestMethod]
public void IsExpireTest()
{
ActivityService service = new ActivityService();
bool actual = service.IsExpire();
Assert.IsFalse(actual); using (ShimsContext.Create())
{
ShimDateTime.NowGet = () => new DateTime(, , );
actual = service.IsExpire();
Assert.IsFalse(actual);
}
}
}
}
5. 然后编写ActivityService类
public class ActivityService
{
public DateTime BeginTime { get; set; } public ActivityService()
{
this.BeginTime = new DateTime(, , ); //仅作演示,无意义
} public bool IsExpire()
{
return BeginTime >= DateTime.Now;
}
}
6. 运行测试通过。然后就可以把实际业务类移动到相应VS项目中,并调整命名空间。
二、Stub
现在假设ActivityService类有一个方法获取是否还能报名,但是它依赖于仓储IActivityRepository(只有遵循依赖反转与接口隔离原则的代码才好使用Stub填充外部依赖)提供的RegisterNumber方法。
1. IActivityRepository接口(新建IRepositories项目并添加该接口)
public interface IActivityRepository
{
/// <summary>
/// 已报名人数
/// </summary>
int RegisterNumber();
}
2. 而我们的单元测试现在不能依赖具体(实际环境中的Repository可能对测试带来影响),这时候就能使用Stub来填充该接口了,添加IRepositories引用,然后与上一个Demo一样的添加IRepositories的Fakes程序集。
3. 在测试类中添加Using代码
using IRepositories;
using IRepositories.Fakes;
4. 编写测试代码
[TestMethod]
public void CanRegisterTest()
{
StubIActivityRepository repository = new StubIActivityRepository();
ActivityService service = new ActivityService(repository); //如果已报名人数小于最多可报名数量则不能再报名,断言CanRegister方法应为True
repository.RegisterNumber = ()=> ;
bool actual = service.CanRegister();
Assert.IsTrue(actual); //如果已报名人数大于等于最多可报名数量则不能再报名,断言CanRegister方法应为False
repository.RegisterNumber = () => ;
actual = service.CanRegister();
Assert.IsFalse(actual);
}
5. ActivityService代码:
public class ActivityService
{
public DateTime BeginTime { get; set; } /// <summary>
/// 最多可报名数量
/// </summary>
private int maxCount = ;
private IActivityRepository repository; public ActivityService()
{
this.BeginTime = new DateTime(, , ); //仅作演示,无意义
} public ActivityService(IActivityRepository repository)
{
// TODO: Complete member initialization
this.repository = repository;
} public bool IsExpire()
{
return BeginTime >= DateTime.Now;
} public bool CanRegister()
{
return repository.RegisterNumber() < this.maxCount;
}
}
总结
stub用于我们可控的代码,shim用于不可控的,例如.NET Framework以及第三方类库等。