学以致用,单元测试的工具越来越多,可是想找一篇如何单元测试的文章却很难。所以偶来写一点自己的心得,也是一步步摸索着。
先从测试一个方法开始,偶随便找了一个小算法。这个方法看来正合适:不能太简单,比如一加一等于二;也不能太复杂,比如牵涉到数据库操作。
这个算法的功能是取得小于Max的N个随机不重复正整数,代码如下:
public static List<Int32> GetRandomNum(int Max, int N)
{
List<int> Source = new List<int>();
List<int> Result = new List<int>();
for (int i = 0; i < Max; i++) Source.Add(i + 1);
for (int n = 0; n < N; n++)
{
int r = new Random().Next(0, Source.Count);
Result.Add(Source[r]); Source.RemoveAt(r);
}
Result.Sort();
return Result;
}
算法很好理解。好久前写的,现在翻出来,一眼看出有问题。不过有点问题正好,看看单元测试能否找出来。
先添一个项目,引用NUnit.Framework,装好NUnit后在Net组件里有,免安装版的在解压后的目录里找。
新建一个Public类用于测试,叫Test,生成随机数的方法的类叫Program。现在无所谓,正式项目里命名得规范一点。Test类加[TestFixture]特性,添加一个Public函数叫做TestGetRandomNum,加上[Test]特性,这样NUnit能认出这个方法是用于测试的方法了。还给该函数加了一个[Category("GetRandomNum")],因为一个功能可能要写几个函数来测试,这个特性表明方法属于测试GetRandom的方法组。
还有几个常用的特性: [TestFixtureUp] [TestFixtureDown] [TearDown] [Setup] [Ignore] [Explicit],据说以后了解这几个差不多了。
接着,第一个测试方法该测试什么呢?以偶写代码的一点经验,每个函数都会把参数验证放在前面,参数无效不往下走了,抛异常什么的该干嘛干嘛,那测试也先从参数测起吧。 现在分析一下参数异常下的输出:若N<1,不管Max值多少都返回空的List;若N>=1,且Max<N,由于Source集合数量不够,应该会引发IndexOutOfRangeException。
写一个测试方法用于测试这两种异常输出,应该写两个测试方法好一点。不过刚入手,一个两个不必分了,写出来好。代码如下:
1 public void TestGetRandomNum3()
2 {
3 int result1 = 0, result2 = 0, result3 = 0;
4 //N<1的case
5 int[,] case1 = { { 0, 0 }, { 9999, 0 }, { -888, 0 }, { 1111, -888 }, { 0, -1 }, { -100, -99 } };
6 for (int i = 0; i < 6; i++)
7 {
8 int Max = case1[i, 0];
9 int N = case1[i, 1];
10 List<int> list = Program.GetRandomNum(Max, N);
11 if( list.Count==0) result1 ++;
12 }
13 //N>=1,Max<N
14 int[,] case2 = { { 0, 5 }, { 0, 1 }, { -10, 1 }, { -10, 30 }, { 10, 20 } };
15 for (int j = 0; j < 5; j++)
16 {
17 int Max = case2[j, 0];
18 int N = case2[j, 1];
19 try
20 {
21 List<int> list = Program.GetRandomNum(Max, N);
22 }
23 catch (IndexOutOfRangeException ex)
24 {
25 result2++;
26 }
27 }
28 //M=N>=1
29 int[,] case3 = { { 1, 1 }, { 10, 10 } };
30 for (int k = 0; k < 2; k++)
31 {
32 int Max = case3[k, 0];
33 int N = case3[k, 1];
34 List<int> list = Program.GetRandomNum(Max, N);
35 bool isMatch = true;
36 for (int n = 0; n < list.Count; n++)
37 {
38 if (n != list[n]) { isMatch = false; break; }
39 }
40 if (isMatch && list.Count == N) result3++;
41 }
42
43 Assert.AreEqual(result1, 6);
44 Assert.AreEqual(result2, 5);
45 Assert.AreEqual(result3, 2);
46 }
运行NUnit加载项目,双击选中刚写的方法,眼前出现一根长长的大红条!