如題,本文主要作為在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(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專案中,並調整名稱空間。
二、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 = ()=> 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以及第三方類庫等。