单元测试又称模块测试,是对软件设计的最小单元的功能、性能、接口和设计约束等的正确性进行检验,检查程序在语法、格式和逻辑上的错误,并验证程序是否符合规范,以发现单元内部可能存在的各种缺陷。
单元测试的对象是软件设计的最小单位——模块、函数或者类。在传统的结构化程序设计语言(如C语言)中,单元测试的对象一般是函数或者过程。在面向对象设计语言(如Java、C#)中,单元测试的对象可以是类,也可以是类的成员函数/方法。由此可见,单元测试与程序设计和编码密切关联,测试者需要根据详细设计说明书和源程序清单来了解模块的I/O条件和逻辑结构。
本节利用风靡全球的“俄罗斯方块游戏排行榜”的程序作为案例来串讲单元测试的内容。
俄罗斯方块游戏(Tetris)的排行榜功能经过编码后,在与其他模块进行集成之前,需要经过单元测试,测试其功能点的正确性和有效性。以便在后续的集成工作中不会引入更多的问题。
俄罗斯方块是一款风靡全球的电视游戏机和掌上游戏机游戏,它由俄罗斯人阿列克谢·帕基特诺夫发明,故得此名。俄罗斯方块的基本规则是移动、旋转和摆放游戏自动输出的各种方块,使之排列成完整的一行或多行并且消除得分。
排行榜功能是俄罗斯方块游戏中不可或缺的一部分,其用于将当前用户的得分与历史得分记录进行比较并重新排序。
该程序主要涉及的功能点有历史记录文件的读取、分数排名的计算与排序、新记录文件的保存、新记录的显示等。这些功能将在一局游戏结束,并获取到该局游戏的得分后启动。
private void gameOver (int score)//游戏结束 {//Display game over string s="您的得分为:"; string al =". char[]A={}; int i=1; _blockSurface.FontStyle= new Font(FontFace,BigFont);//设置基本格式_blockSurface.FontFormat.Alignment = StringAlignment.Near; _blockSurface.DisplavText ="GAME OVER!! string sc = Convert.ToString( score);//得到当前玩家的分数//write into file; string path="D: test2.txt";//文件路径 try{ FileStream fs = newFileStream (path,FileMode.OpenOrCreate,FileAccess.ReadWrite); StreamReader strmreader = new StreamReader(fs);//建立读文件流 String[] str = new String[5]; String[] split = new String[5]; while(strmreader.Peek()!= -1) for(i=0;i<5;i++) strli] = strmreader.ReadLine();//以行为单位进行读取,赋予数组 //str[i] split [i]split[i]= str[i].split(':')[1];//按照":"将文字分开,赋予数组 person1 = Convert.ToInt32(split[0]); //splt[0]的值赋予第一名 person2 = Convert.ToInt32(split[1]); //splt[1]的值赋予第一名 person3 = Convert.ToInt32(split[2]); //splt[2]的值赋予第一名 person4 = Convert.ToInt32(split[3]); //splt[3]的值赋予第一名 person5 = Convert.ToInt32(split[4]); //splt[4]的值赋予第一名 strmreader.Close(); //关闭流 fs.Close(); FileStream ffs = new FileStream(path, FileMode.OpenOrCreate, FileAccess.ReadWrite)); StreamWriter sw = new StreamWriter(ffs)//建立写文件流 if(_score > person1)//如果当前分数大于第一名,排序 { person5=person4; person4 = person3; person3 = person2; person2 = person1; person1=score; } else if(_score> person2)//如果当前分数大于第二名,排序 { person5 = person4; person4 = person3; person3 = person2; person2= _score; } else if(_score> person3)//如果当前分数大于第三名,排序 { person5 = person4; person4 = person3; person3 =_score; else if(_score> person4)//如果当前分数大于第四名,排序 { person5 = person4; person4 =_score; } else if(_score > person5)//如果当前分数大于第五名,排序 { person5 = _score; //在文件中的文件内容 string ppl ="第一名:“+ Convert.ToString(person1); string pp2 ="第二名:"+ Convert.ToString(person2); string pp3 ="第三名:”+ Convert.ToString(person3); string pp4 ="第四名:"+ Convert.ToString(person4); string pp5 ="第五名"+ Convert.ToString(person5); string ppR= pp1+"r\n"+ pp2 +"r\n" + pp3 +"r\n"+ pp4+"r\n"+ pp5 +"r\n"; byte[] info = new UTF8Encoding(true).GetBytes(ppR); sw.Write(ppR); //将内容写入文件 sw.Close(); ffs.Close(); } Catch(Exception ex) { Console.WriteLine(ex.ToString()); } S=s+""+ sC; //Draw surface to display text; MessageBox.Show(s); //在界面中显示排行榜内容 }
下面将利用相关静态和动态(白盒测试、黑盒测试)方法对案例进行相应的测试,得到测试报告与错误列表,在实际项目中可进一步反馈给开发方进行Bug的确认与修复。
利用代码走查的方法检查该模块的代码,对代码质量进行初步评估。具体实现如表1所示。
■ 表1 代码走查情况记录
从表1的分析中可以看出,本模块的代码基本情况如下。
(1) 代码直观。
(2) 代码和设计文档对应。
(3) 无用的代码已经被删除。
(4) 注释过于简单。
基本路径测试法是在程序控制流图的基础上,通过分析控制构造的环路复杂性,导出可执行的路径集合,从而设计测试用例的方法。首先需要简化程序模块,绘制程序模块如图2所示。接着按照模块图的设计路径来覆盖策略。主要可分为以下4步执行。
■ 图2 程序模块图
1) 绘制程序的控制流图
基本路径测试法的第一步是绘制控制流图,根据程序模块图的逻辑关系,获得该程序块的控制流图,如图3所示。
■ 图3 程序模块的控制流图
2) 计算环路复杂度
其次是根据控制流图计算环路复杂度,环路复杂度是一种为程序逻辑复杂性提供定量测度的软件度量,该度量将用于计算程序基本的独立路径数目,为确保所有语句至少执行一次的测试数量的上界。
V(G)=P+1=5+1=6
根据以上公式确定至少要覆盖6条路径。
3) 导出独立路径
根据控制流图可以方便地得到以下6条路径。
path1:1—2—11。
path2:1—3—4—11。
path3:1—3—5—6—11。
path4:1—3—5—7—8—11。
path5:1—3—5—7—9—10—11。
path6:1—3—5—7—9—11。
4) 设计测试用例
最后设定一组初始参数,以此来设计测试用例。令:
person1=23
person2=20
person3=10
person4=6
person5=4
作为测试输入,可设计测试用例如表2所示。
■ 表2 基本路径法测试用例
3. 边界值分析
边界值分析法利用输入变量的最小值、略大于最小值、输入范围内任意值、略小于最大值、最大值等来设计测试用例。
由于输入的只会是数据,且数据均大于0,因此可令:
person1 = 23
person2 = 20
person3 = 10
person4 = 6
person5 = 4
采用边界值法设计测试用例如表3所示。
■ 表3 边界值法测试用例
将设计的测试用例整理合并为测试用例集合,必要时需要开发相应的驱动模块和桩模块。本次测试需要开发一个驱动模块,用于初始化相应的参数,并调用待测模块以达到测试效果。驱动模块代码如下。
import java.io.BufferedReader; import java.io.IOException; import java.io.InputStreamReader; public class Main(){ public static void main(String[]args) throws IOException{ int person1 = 23,person2 = 20, person3 = 10,person4 = 6,person5 = 4; int score; String s; BufferedReader bf = new BufferedReader(newInputStreamReader(System.in)); s = bf.readLine(): score = Integer.valueOf(s); _gameOver(score); } }
测试结果可利用Bug记录平台进行记录,在实际项目中则可反馈给开发人员,由开发人员确认并修复。
测试结束后,形成测试报告。