软件测试技术与质量保证主讲人:徐丽4.7可测性4.7.1要有一套统一打印函数及详细的说明在同一项目组或产品组内,要有一套统一的为集成测试与系统联调准备的调测开关及相应打印函数,并且要有详细的说明。说明:本规则是针对项目组或产品组的。4.7.2信息串的格式要统一在同一项目组或产品组内,调测打印出的信息串的格式要有统一的形式。信息串中至少要有所在模块名(或源文件名)及行号。说明:统一的调测信息格式便于集成测试。4.7.3选择恰当的测试点编程的同时要为单元测试选择恰当的测试点,并仔细构造测试代码、测试用例,同时给出明确的注释说明。测试代码部分应作为(模块中的)一个子模块,以方便测试代码在模块中的安装与拆卸(通过调测开关)。说明:为单元测试而准备。4.7.4集成测试/系统联调之前的准备在进行集成测试/系统联调之前,要构造好测试环境、测试项目及测试用例,同时仔细分析并优化测试用例,以提高测试效率。说明:好的测试用例应尽可能模拟出程序所遇到的边界值、各种复杂环境及一些极端情况等。4.7.5使用断言来发现软件问题使用断言来发现软件问题,提高代码可测性。说明:断言是对某种假设条件进行检查(可理解为若条件成立则无动作,否则应报告),它可以快速发现并定位软件问题,同时对系统错误进行自动报警。断言可以对在系统中隐藏很深,用其它手段极难发现的问题进行定位,从而缩短软件问题定位时间,提高系统的可测性。实际应用时,可根据具体情况灵活地设计断言。4.7.6使用断言检查非法情况用断言来检查程序正常运行时不应发生但在调测时有可能发生的非法情况。4.7.7断言的正确使用不能用断言来检查最终产品肯定会出现且必须处理的错误情况。说明:断言是用来处理不应该发生的错误情况的,对于可能会发生的且必须处理的情况要写防错程序,而不是断言。如某模块收到其它模块或链路上的消息后,要对消息的合理性进行检查,此过程为正常的错误检查,不能用断言来实现。4.7.8对较复杂的断言加上明确的注释说明:为复杂的断言加注释,可澄清断言含义并减少不必要的误用。4.7.9用断言确认函数的参数示例:假设某函数参数中有一个指针,那么使用指针前可对它检查,如下。intExamFun(unsignedchar*str){EXAM_ASSERT(str!=NULL);//用断言检查“假设指针不为空”这个条件...//otherprogramcode}4.7.10确保不使用没有定义的特性或功能用断言保证没有定义的特性或功能不被使用。示例:假设某通信模块在设计时,准备提供“无连接”和“连接”这两种业务。但当前的版本中仅实现了“无连接”业务,且在此版本的正式发行版中,用户(上层模块)不应产生“连接”业务的请求,那么在测试时可用断言检查用户是否使用“连接”业务。#defineEXAM_CONNECTIONLESS0//无连接业务#defineEXAM_CONNECTION1//连接业务intMsgProcess(EXAM_MESSAGE*msg){unsignedcharservice;/*messageserviceclass*/EXAM_ASSERT(msg!=NULL);service=GetMsgServiceClass(msg);EXAM_ASSERT(service!=EXAM_CONNECTION);//假设不使用连接业务...//otherprogramcode}4.7.11用断言对程序开发环境的假设进行检查用断言对程序开发环境(OS/Compiler/Hardware)的假设进行检查。说明:程序运行时所需的软硬件环境及配置要求,不能用断言来检查,而必须由一段专门代码处理。用断言仅可对程序开发环境中的假设及所配置的某版本软硬件是否具有某种功能的假设进行检查。如某网卡是否在系统运行环境中配置了,应由程序中正式代码来检查;而此网卡是否具有某设想的功能,则可由断言来检查。对编译器提供的功能及特性假设可用断言检查,原因是软件最终产品(即运行代码或机器码)与编译器已没有任何直接关系,即软件运行过程中(注意不是编译过程中)不会也不应该对编译器的功能提出任何需求。示例:用断言检查编译器的int型数据占用的内存空间是否为2,如下。EXAM_ASSERT(sizeof(int)==2);4.7.12正式软件产品中应把断言及其它调测代码去掉正式软件产品中应把断言及其它调测代码去掉(即把有关的调测开关关掉)。说明:加快软件运行速度。4.7.13不能影响软件实现的功能在软件系统中设置与取消有关测试手段,不能对软件实现的功能等产生影响。说明:即有测试代码的软件和关掉测试代码的软件,在功能行为上应一致。4.7.14减少维护的难度用调测开关来切换软件的DEBUG版和正式版,而不要同时存在正式版本和DEBUG版本的不同源文件,以减少维护的难度。4.7.15确保软件版本在实现功能上的一致性软件的DEBUG版本和发行版本应该统一维护,不允许分家,并且要时刻注意保证两个版本在实现功能上的一致性。4.7.16编写代码之前要注意的事项在编写代码之前,应预先设计好程序调试与测试的方法和手段,并设计好各种调测开关及相应测试代码如打印函数等。说明:程序的调试与测试是软件生存周期中很重要的一个阶段,如何对软件进行较全面、高率的测试并尽可能地找出软件中的错误就成为很关键的问题。因此在编写源代码之前,除了要有一套比较完善的测试计划外,还应设计出一系列代码测试手段,为单元测试、集成测试及系统联调提供方便。4.7.17调测开关应分为不同级别和类型调测开关应分为不同级别和类型。说明:调测开关的设置及分类应从以下几方面考虑:针对模块或系统某部分代码的调测;针对模块或系统某功能的调测;出于某种其它目的,如对性能、容量等的测试。这样做便于软件功能的调测,并且便于模块的单元测试、系统联调等。4.7.18用断言宣布发生错误编写防错程序,然后在处理错误之后可用断言宣布发生错误。4.8程序效率4.8.1编程时要经常注意代码的效率说明:代码效率分为全局效率、局部效率、时间效率及空间效率。全局效率是站在整个系统的角度上的系统效率;局部效率是站在模块或函数角度上的效率;时间效率是程序处理输入任务所需的时间长短;空间效率是程序所需内存空间,如机器代码空间大小、数据空间大小、栈空间大小等。4.8.2提高代码效率在保证软件系统的正确性、稳定性、可读性及可测性的前提下,提高代码效率。说明:不能一味地追求代码效率,而对软件的正确性、稳定性、可读性及可测性造成影响。4.8.3局部效率与全局效率局部效率应为全局效率服务,不能因为提高局部效率而对全局效率造成影响。4.8.4提高空间效率通过对系统数据结构的划分与组织的改进,以及对程序算法的优化来提高空间效率。说明:这种方式是解决软件空间效率的根本办法。示例:如下记录学生学习成绩的结构不合理。typedefunsignedcharBYTE;typedefunsignedshortWORD;typedefstructSTUDENT_SCORE_STRU{BYTEname[8];BYTEage;BYTEsex;BYTEclass;BYTEsubject;floatscore;}STUDENT_SCORE;因为每位学生都有多科学习成绩,故如上结构将占用较大空间。应如下改进(分为两个结构),总的存贮空间将变小,操作也变得更方便。typedefstructSTUDENT_STRU{BYTEname[8];BYTEage;BYTEsex;BYTEclass;}STUDENT;typedefstructSTUDENT_SCORE_STRU{WORDstudentIndex;BYTEsubject;floatscore;}STUDENT_SCORE;4.8.5循环体内工作量最小化说明:应仔细考虑循环体内的语句是否可以放在循环体之外,使循环体内工作量最小,从而提高程序的时间效率。示例:如下代码效率不高。for(ind=0;indMAX_ADD_NUMBER;ind++){sum+=ind;backSum=sum;/*backupsum*/}4.8.6算法的优化仔细分析有关算法,并进行优化。4.8.7改进系统及模块处理输入的方式仔细考查、分析系统及模块处理输入(如事务、消息等)的方式,并加以改进。4.8.8提高程序效率对模块中函数的划分及组织方式进行分析、优化,改进模块中函数的组织结构,提高程序效率。说明:软件系统的效率主要与算法、处理任务方式、系统功能及函数结构有很大关系,仅在代码上下功夫一般不能解决根本问题。4.8.9留心代码效率编程时,要随时留心代码效率;优化代码时,要考虑周全。4.8.10恰当优化代码提高效率不应花过多的时间拼命地提高调用不很频繁的函数代码效率。说明:对代码优化可提高效率,但若考虑不周很有可能引起严重后果。4.8.11慎重使用汇编嵌入方式要仔细地构造或直接用汇编编写调用频繁或性能要求极高的函数。说明:只有对编译系统产生机器码的方式以及硬件系统较为熟悉时,才可使用汇编嵌入方式。嵌入汇编可提高时间及空间效率,但也存在一定风险。4.8.12提高空间效率在保证程序质量的前提下,通过压缩代码量、去掉不必要代码以及减少不必要的局部和全局变量,来提高空间效率。说明:这种方式对提高空间效率可起到一定作用,但往往不能解决根本问题。4.8.13在多重循环中,应将最忙的循环放在最内层说明:减少CPU切入循环层的次数。示例:如下代码效率不高。for(row=0;row100;row++){for(col=0;col5;col++){sum+=a[row][col];}}可以改为如下方式,以提高效率。for(col=0;col5;col++){for(row=0;row100;row++){sum+=a[row][col];}}4.8.14尽量减少循环嵌套层次4.8.15避免循环体内含判断语句避免循环体内含判断语句,应将循环语句置于判断语句的代码块之中。说明:目的是减少判断次数。循环体中的判断语句是否可以移到循环体外,要视程序的具体情况而言,一般情况,与循环变量无关的判断语句可以移到循环体外,而有关的则不可以。示例:如下代码效率稍低。for(ind=0;indMAX_RECT_NUMBER;ind++){if(dataType==RECT_AREA){areaSum+=rectArea[ind];}else{rectLengthSum+=rect[ind].length;rectWidthSum+=rect[ind].width;}}因为判断语句与循环变量无关,故可如下改进,以减少判断次数。if(dataType==RECT_AREA){for(ind=0;indMAX_RECT_NUMBER;ind++){areaSum+=rectArea[ind];}}else{for(ind=0;indMAX_RECT_NUMBER;ind++){rectLengthSum+=rect[ind].length;rectWidthSum+=rect[ind].width;}}4.8.16用乘法或其它方法代替除法尽量用乘法或其它方法代替除法,特别是浮点运算中的除法。说明:浮点运算除法要占用较多CPU资源。示例:如下表达式运算可能要占较多CPU资源。#definePAI3.1416radius=circleLength/(2*PAI);应如下把浮点除法改为浮点乘法。#definePAI_RECIPROCAL(1/3.1416)//编译器编译时,将生成具体浮点数radius=circleLength*PAI_RECIPROCAL/2;4.8.17不要一味追求紧凑的代码说明:因为紧凑的代码并不代表高效的机器码。4.9宏4.9.1用宏定义表达式时,要使用完备的括号示例:如下定义的宏都存在一定的风险。#defineRECTANGLE_AREA(a,b)a*b#defineRECTANGLE_AREA(a,b)(a*b)#defineRECTANGLE_AREA