VS2012 Unit Test——Microsoft Fakes入门

如题,本文主要作为在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以及第三方类库等。

posted @ 2013-10-06 16:41 dong.net 阅读(...) 评论(...) 编辑 收藏