.net 软件测试自动化之道

 

 

API(Application Programming Interface)包括单元测试(Unit Testing),模块测试(Module Testing),组件测试(Component Testing ),元件测试(Element Testing)

为了将测试的程序和用于测试他们的测试套件(test harness)区分,待测程序叫做SUT(System Under Test),AUT(Application Under Test)或IUT(Implementation Under Test)

手工测试这个API包括:
创建一个小的测试程序,把Methods类拷贝到测试程序,针对待测试方法硬编码(hard-coding)输入一些值,运行这个存根程序(stub program)得到实际输出结果,然后比较实际结果和预期结果是否一致。把测试结果记录到excel表格或类似数据存储文件。

自动话测试的优势:

速度:可以非常快速的运行成千上万个测试用例
精确性:不受人为因素干扰,可以记录测试结果
确定性:每次以同样的方式运行
效率:时间不受限制
提高技能:提高测试人员的兴趣还有自身技能

如下流程,很值得学习

1.1创建文本文件以及数据
如何在简单的文本文档创建并存储用于API测试用例的数据(可以独立于测试套件存在[比较倾向于这个方式],也可以嵌入测试套件)
设计:使用逗号作为分隔符的文本文件,包含唯一测试用例Id,一个或多个输入值和期望结果
方案:
001:ArithmeticMean:2 4 8:4.6777
002:ArithmeticMean:1 5:3.000
003:ArithmeticMean:1 5:6.00000:deliberate failure

每个测试用例有4个字段(用":"分开),测试用例ID,待测方法,测试用例输入(由空格分开)以及期望结果

1.2使用数据
如何从测试用例文件读入每条测试用例
设计:通过while循环遍历测试用例文件的每一行,使用system.IO.StreamReader对象读入测试用例

  //使用测试数据
        public void UseData()
        {
            FileStream fs=new FileStream("..\\TestCases.txt",FileMode.Open);
            StreamReader sr=new StreamReader(fs);
            string lines;
            while ((lines = sr.ReadLine()) != null) ;
            {
                //解析每个测试用例
                //调用待测方法
                //判断是否通过
                //记录测试用例结果
            }
            sr.Close();
            fs.Close();
        }

 

1.3解析测试用例
如何解析出由字符分隔开的测试用例的各个字段

设计:使用string.Split()方法,把分隔符作为输入传给他,然后把返回值存入一个字符数组
方案:

 while ((lines = sr.ReadLine()) != null) ;
            {
                //解析每个测试用例
                tokens = lines.Split(':');
                caseID = tokens[0];
                methd = tokens[1];
                tempInput = tokens[2].Split(' ');
                expected = tokens[3];
                ////使用多个分隔符的时候,可以创建包含分隔符的数组,然后把这个数组穿给Split()
                //char[] separators=new char['#','!',','];
                //string parts = line.Split(separators);

                //调用待测方法
                //判断是否通过
                //记录测试用例结果
            }

 

1.4把数据转换为合适的类型

如何把测试用例的输入数据或者期望结果从string类型转为其他数据类型
设计:通过选用合适的静态Parese()方法,实施显式类型转换

int[] input = new int[tempInput.Length];
                for (int i=0;i<input.Length;i++)
                {
                    input[i] = int.Parse(tempInput[i]);//Parse()接受一个string作为参数,并且返回调用者所用的数据类型
                }

 

1.5判定测试用例通过与否
如何判定API测试用例是通过还是失败
设计:调用待测方法,传给他测试用例的输入,得到返回值。然后比较实际结果与测试用例中读入的期望结果是否一致

double actual = 0.0;
                if (method=="ArithmeticMean")
                {
                    actual = MathLib.Methods.ArithmeticMean(input);
                    if (actual.ToString("F4") == expected)
                    {
                        Console.WriteLine(caseID+"Pass");
                    }
                    else
                    {
                        Console.WriteLine(caseID+"*Fail*"+method+"actual="+actual.ToString("F4")+"expected="+expected);
                    }
                }
                else
                {
                    Console.WriteLine("Method not recognized");
                }

测试API方法的时候,必须考虑到待测方法是无状态(stateless)还是有状态的(stateful).大部分API是无状态的(也就是调用之间是相互独立的)

测试套件必须能访问待测的API方法,大多数情况下把这些API方法的DLL作为工程引用添加到待测程序中;有些情况下,需要把待测方法的代码copy到测试套件

 

1.6记录测试结果

如何把测试用例的结果存入独立于测试程序的简单文本文件

设计:使用System.IO.StreamWriter对象,把测试用例ID和测试结果写到一个文本文件

FileStream ofs=new FileStream("..\\TestResults.txt",FileMode.CreateNew);//如果存在TestResults.txt,会抛出异常,
                //可以使用时间戳加在测试结果的文件名称里;还可以使用的策略,将测试结果的文件名作为参数传进来,手动产生新的文件名
                StreamWriter owr=new StreamWriter(ofs);
                while ((lines=sr.ReadLine())!=null)
                {
                    //对lines 进行解析
                    if (method=="ArithmeticMean")
                    {
                        actual = MathLib.Methods.ArithmeticMean(input);
                        if (actual.ToString("F4")==expected)
                        {
                            //同时将结果写入命令行窗口和txt文件
                            Console.WriteLine(caseID+"Pass");
                            owr.WriteLine(caseID+"Pass");
                        }
                        else
                        {
                            owr.WriteLine(caseID+"*Fali*");
                        }
                    }
                    else
                    {
                        owr.WriteLine(caseID+"Unknow method");
                    }
                }
                //如果捕获到异常,记得在cath/finally关掉已经打开的输入输出流
                //catch(Exception ex){console.writeline("Unexpected fatal error"+ex.Message)};owr.Close();
                owr.Close();
                ofs.Close();
            }

1.7给测试用例加上时间戳

如何给测试用例加上时间戳,以便区分不同批次的运行结果

设计:把DateTime.Now属性作为参数传给CreateDirectory(),也可以传给FileStream()构造函数

 string stamp = DateTime.Now.ToString("t");
                stamp = stamp.Replace(":","-");
                string path1 = "..\\TestResluts-" + stamp + ".txt";
                FileStream ofs1=new FileStream(path1,FileMode.Create);//TestResults-xxx.txt(只有时间);将"t"改成"s"   年-月-日T
                StreamWriter owr=new StreamWriter(ofs);

 

1.8通过计算对测试结果进行总结

如何计算测试用例的结果以及追踪通过的pass用例和失败用例的个数
设计:使用简单的整数计数器,每次开始运行测试的时候,测试器初始化为0

 

 

              int numPass = 0, numFail = 0;
                while ((lines = sr.ReadLine()) != null)
                {
                    //对lines 进行解析
                    if (method == "ArithmeticMean")
                    {
                        actual = MathLib.Methods.ArithmeticMean(input);
                        if (actual.ToString("F4") == expected)
                        {
                            //同时将结果写入命令行窗口和txt文件
                            //Console.WriteLine(caseID + "Pass");
                            owr.WriteLine(caseID + "Pass");
                            ++numPass;
                        }
                        else
                        {
                            owr.WriteLine(caseID + "*Fali*");
                            ++numFail;
                        }
                    }
                    else
                    {
                        owr.WriteLine(caseID + "Unknow method");
                    }
                }
                Console.WriteLine("Number cases passed ="+numPass);
                Console.WriteLine("Number cases failed ="+numFail);
                Console.WriteLine("Total cases ="+(numFail+numPass));
                double percent=((double)numPass)/(numPass+numFail);
                Console.WriteLine("Percent passed ="+percent.ToString("p"));