如题,本文主要作为在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进行单元测试的一起交流进步。
以下将模拟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(2014, 5, 5); actual = service.IsExpire(); Assert.IsFalse(actual); } } } }
5. 然后编写ActivityService类
public class ActivityService { public DateTime BeginTime { get; set; } public ActivityService() { this.BeginTime = new DateTime(2014, 3, 3); //仅作演示,无意义 } public bool IsExpire() { return BeginTime >= DateTime.Now; } }
6. 运行测试通过。然后就可以把实际业务类移动到相应VS项目中,并调整命名空间。
现在假设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 = ()=> 20; bool actual = service.CanRegister(); Assert.IsTrue(actual); //如果已报名人数大于等于最多可报名数量则不能再报名,断言CanRegister方法应为False repository.RegisterNumber = () => 50; actual = service.CanRegister();
Assert.IsFalse(actual);
}
5. ActivityService代码:
public class ActivityService { public DateTime BeginTime { get; set; } /// <summary> /// 最多可报名数量 /// </summary> private int maxCount = 50; private IActivityRepository repository; public ActivityService() { this.BeginTime = new DateTime(2014, 3, 3); //仅作演示,无意义 } 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以及第三方类库等。