本文主要介绍Visual Studio(2012+)单元测试框架的一些技巧:

  1. 如何模拟类的静态构造函数
  2. 如何测试某方法被调用过
  3. 如何测试某方法执行的次数
  4. 并行编程测试注意事项

一、如何模拟类的静态构造函数

1.1 被测代码

namespace BlogDemo.UTDemo.Tricks
{
public class StaticConstructorExample
{
static StaticConstructorExample()
{
//step1:read data from web service
//step2:save data into database
} public int DoSomething()
{
return ;
}
}
}

上面的类有一个静态构造函数,里面做两件事情:

  1. 从某webservice读取数据
  2. 将读取到的数据保存到数据库。

以上两件事情都是对外界环境的严重依赖

我们需要测试的方法是DoSomething,需要测试这个类,就需要实例化StaticConstructorExample这个类,实例化这个类之前,其静态构造函数将会被自动执行:这个点就是问题。

  1. 静态构造函数依赖外部环境,这个外部环境有可能在某一天就挂了
  2. 如果外部环境挂了,静态构造函数就会报错,那我们的单元测试就无法按照预期进行运行
  3. DoSOmething本身不直接依赖外部环境,但是通过静态构造函数间接依赖了外部环境,这违反了单元测试repeatable原则(当外部环境挂了,我们的测试就无法运行)

解决办法依然是Shim,通过ms.test的Shim可以模拟一个类的静态构造函数,从而改变静态构造函数的行为,消除外部依赖。

1.2 测试代码

 [TestClass]
public class StaticConstructorExampleTests
{
[TestMethod]
public void DoSomethingTest()
{
using (ShimsContext.Create())
{
var isStaticConstructorExecuted = false;
ShimStaticConstructorExample.StaticConstructor = () =>
{
//step1:mock web service
//step2:mock database
isStaticConstructorExecuted = true;
};
var example = new StaticConstructorExample();
var data = example.DoSomething();
Assert.AreEqual(, data);
Assert.IsTrue(isStaticConstructorExecuted);
} }
}

上面的代码是通过ShimClassName.StaticConstructor来实现对静态构造函数进行模拟的。

二、如何测试某方法被调用过

2.1 被测代码

基础代码,在DemoClass中进行调用

    public interface IGoodActionHandler
{
void Action();
} public interface IBadActionHandler
{
void Action();
} public class GoodHandler : IGoodActionHandler
{
public void Action()
{
throw new NotImplementedException();
}
} public class BadHandler : IBadActionHandler
{
public void Action()
{
throw new NotImplementedException();
}
}

要被测试的类:

  public class DemoClass
{
IGoodActionHandler goodHandler;
IBadActionHandler badHandler;
public DemoClass()
{
this.goodHandler = new GoodHandler();
this.badHandler = new BadHandler();
}
public void DoSomething(int type)
{
if (type == )
{
DoSomethingGood();
}
else
{
DoSomethingBad();
}
} private void DoSomethingGood()
{
this.goodHandler.Action();
} private void DoSomethingBad()
{
this.badHandler.Action();
}
}

要测试上面的DoSomething方法,上面DoSomethingGood和DoSomethingBad都依赖自接口,在测试的时候都可以进行mock(stub)。在侧测DoSomething方法是只需要验证当Type为1时执行了goodHandler的action,否则执行badHandler的Action。

这也是单元测试的一个关键点:关注单元。这里不关注goodhandler 和badhandler的内部逻辑(这两个handler的内部逻辑可以单独测试,属于另外的单元),这里只关注是否按照逻辑路由到了正确的handler。

2.2测试代码

  [TestClass]
public class DemoClassTests
{
[TestMethod]
public void DoSomething_DoSomethingGood_Tests()
{
var goodHandlerExecuted = false;
StubIGoodActionHandler goodHandler = new StubIGoodActionHandler()
{
Action = () =>
{
goodHandlerExecuted = true;//如果执行了goodhandler则置为true
}
};
var badHandlerExecuted = false;
StubIBadActionHandler badHandler = new StubIBadActionHandler()
{
Action = () =>
{
badHandlerExecuted = true;//如果执行了badhandler则置为true
}
};
var demoClass=new DemoClass();
demoClass.BadHandler=badHandler;//注入badhandler
demoClass.GoodHandler=goodHandler;//注入goodhandler
demoClass.DoSomething();
Assert.IsTrue(goodHandlerExecuted);//执行了goodhandler
Assert.IsFalse(badHandlerExecuted);//没有执行badhandler
}
}

上面使用了两个技术

  1. 面向接口编程,使用Stub技术,动态注入Stub(打桩),在测试的时候对对象进行了模拟
  2. 改变了Stub对象的行为,本文只是通过在Stub对象的内部设置标志位的值来表示是否执行了这一个步骤。

上面的标志位的方法Shim技术照样使用,静态构造函数的模拟就是通过Shim后在构造函数内部修改标识位来验证构造函数被执行的。接下来介绍的测试执行次数也是通过shim后修改标识位来实现的

三、如何测试某方法执行的次数

3.1 被测代码

  public class DemoClass
{
private LoopHandler handler;
public DemoClass()
{
handler = new LoopHandler();
}
public void DoSomething(int times)
{
for (var i = ; i < times; i++)
{
handler.Do();
}
}
} public class LoopHandler
{
public void Do() { }
}

现在要验证DoSomething是否执行了times次Do方法。

3.2 测试代码

 [TestMethod]
public void DoSomething_RunTimes_Test()
{
using(ShimsContext.Create())
{
var times = ;
ShimLoopHandler.AllInstances.Do = (@this) =>
{
times++;//每进入一次加一次
};
var givenTimes = ;
new DemoClass().DoSomething(givenTimes);
Assert.AreEqual(times, givenTimes);
}
}

思想和上面Stub是一样,在Shim模拟的方法内部进行计数器相加。

3.3 并行编程测试注意事项

这里有一点需要注意一下,因为int的++不是线程安全的,如果把DemoClass的循环修改成多个线程并行执行的话测试代码需要做相应的调整。,对times++进行lock。

 public void DoSomething(int times)
{
Parallel.For(, times, (item) =>//并行执行
{
handler.Do();
});
} [TestMethod]
public void DoSomething_RunTimes_Test()
{
using(ShimsContext.Create())
{
var times = ;
ShimLoopHandler.AllInstances.Do = (@this) =>
{
lock (this)//lock线程安全
{
times++;//每进入一次加一次
}
};
var givenTimes = ;
new DemoClass().DoSomething(givenTimes);
Assert.AreEqual(times, givenTimes);
}
}

上面的代码当Times不是特别大的时候一般不加lock也是可以的,不是特别大的时候一般不会产生线程安全问题。但是当times比较大的时候不加lock就会出问题。

当times是100000时去掉lock。我运行就出现了问题:

测试就是为了严谨,如果有上面并行的问题,测试的时候还是加上lock比较好。

使用Visual Studio进行单元测试-Part5的更多相关文章

  1. 【MVC 4】4.MVC 基本工具(Visual Studio 的单元测试、使用Moq)

     作者:[美]Adam Freeman      来源:<精通ASP.NET MVC 4> 3.Visual Studio 的单元测试 有很多.NET单元测试包,其中很多是开源和免费的.本 ...

  2. [转]Visual Studio 2010 单元测试目录

    Visual Studio 2010 单元测试共分七个部分: 普通单元测试.顺序单元测试.压力测试,Generic测试.数据库测试.UI界面测试和Web性能测试. 这个系列的博客实例程序可以在下面的链 ...

  3. 使用Visual Studio进行单元测试

    一.使用Visual Studio进行单元测试的几个建议 1.先写单元测试(依我愚见,应该是接口先行,如果有的话) -> 测试失败 -> 以最小的改动(即编写实际代码)使测试通过(而在VS ...

  4. Visual Studio 2010 单元测试目录

    单元测试的重要性这里我就不多说了,以前大家一直使用NUnit来进行单元测试,其实早在Visual Studio 2005里面,微软就已经集成了一个叫Test的专门测试插件,经过几年的发展,这个工具现在 ...

  5. Visual Studio 2010 单元测试之一---普通单元测试

    原文:Visual Studio 2010 单元测试之一---普通单元测试 本文以Visual Studio 2010为例,来介绍如何在Visual Studio里面进行单元测试. 首先来介绍普通单元 ...

  6. Visual Studio 2010 单元测试--运行测试并查看代码覆盖率

    原文:Visual Studio 2010 单元测试--运行测试并查看代码覆盖率 运行测试并查看代码覆盖率对程序集中的代码运行测试时,可以通过收集代码覆盖率数据来查看正在测试的项目代码部分. 运行测试 ...

  7. MVC 基本工具(Visual Studio 的单元测试、使用Moq)

    3.Visual Studio 的单元测试 有很多.NET单元测试包,其中很多是开源和免费的.本文打算使用 Visual Studio 附带的内建单元测试支持,但其他一些.NET单元测试包也是可用的. ...

  8. [转]Visual Studio 2010单元测试(2)--运行测试并查看代码覆盖率

    Visual Studio 2010 单元测试--运行测试并查看代码覆盖率 运行测试并查看代码覆盖率对程序集中的代码运行测试时,可以通过收集代码覆盖率数据来查看正在测试的项目代码部分. 运行测试并查看 ...

  9. C# Note31: 如何使用Visual Studio做单元测试

    待更! 使用Visual Studio 2013进行单元测试--初级篇 带你玩转Visual Studio——单元测试(C++例)

随机推荐

  1. Collective Mindsets (medium) (逻辑题)

    B - Collective Mindsets (medium) Time Limit:1000MS     Memory Limit:262144KB     64bit IO Format:%I6 ...

  2. ubuntu14.04 desktop 32-bit kvm装windows xp

    经过这几天来的折腾,总算是在ubuntu14.04用kvm装上了xp, 看不少的的贴,也绕了不少的圈,总的来说,非常感谢CSDN上的"上善若水75",看着他写的一个分类" ...

  3. java Filter的简单使用

    java web中的过滤器的简单使用.直接上代码.1.web.xml <?xml version="1.0" encoding="UTF-8"?> ...

  4. IOS navigationItem 设置返回button,title图片和rightBarButtonItem

    1.自己定义返回button UIBarButtonItem *backItem = [[UIBarButtonItem alloc] initWithTitle:@"返回" st ...

  5. host更新

    http://alsohosts.herokuapp.com/ google镜像站https://goge.ml/

  6. LeetCode:验证回文串【125】

    LeetCode:验证回文串[125] 题目描述 给定一个字符串,验证它是否是回文串,只考虑字母和数字字符,可以忽略字母的大小写. 说明:本题中,我们将空字符串定义为有效的回文串. 示例 1: 输入: ...

  7. CentOS取消屏幕保护自动锁屏功能

    CentOS系统在用户闲置一段时间(默认为5分钟)后,会启动屏幕保护程序(默认的屏保为黑屏),并要求重新输入密码才能回到原来的桌面. 设置屏幕保护:System -> Preferences - ...

  8. ZOJ - 3705 Applications 【模拟】

    题目链接 http://acm.zju.edu.cn/onlinejudge/showProblem.do?problemCode=3705 题意 给出N个队员 然后一个教练要从中选择 M名队员 要选 ...

  9. 简介windows的环境变量

    环境变量一般是指在操作系统中用来指定操作系统运行环境的一些参数,比如临时文件夹位置和系统文件夹位置等.这点有点类似于DOS时期的默认路径,当你运行某些程序时除了在当前文件夹中寻找外,还会到设置的默认路 ...

  10. 第十二、模块二、调用中国天气网和qqOnline及TrainTimeWebService接口来突出Json方法

    一. 浏览网页的时候,发送的请求.服务器反回来的永远是字符串,由于服务器后台使用的语言不通,所以就需要用工具反解,这里用到了json json方法一 json.loads()将字符串转化为python ...