软件工程课本讲解第4章程序编码.ppt

上传人:小飞机 文档编号:6610934 上传时间:2023-11-17 格式:PPT 页数:115 大小:393KB
返回 下载 相关 举报
软件工程课本讲解第4章程序编码.ppt_第1页
第1页 / 共115页
软件工程课本讲解第4章程序编码.ppt_第2页
第2页 / 共115页
软件工程课本讲解第4章程序编码.ppt_第3页
第3页 / 共115页
软件工程课本讲解第4章程序编码.ppt_第4页
第4页 / 共115页
软件工程课本讲解第4章程序编码.ppt_第5页
第5页 / 共115页
点击查看更多>>
资源描述

《软件工程课本讲解第4章程序编码.ppt》由会员分享,可在线阅读,更多相关《软件工程课本讲解第4章程序编码.ppt(115页珍藏版)》请在三一办公上搜索。

1、程序编码,编码阶段的任务结构化程序设计 程序设计风格 程序效率 程序复杂性度量,做为软件工程过程的一个阶段,程序编码是设计的继续。程序设计语言的特性和程序设计风格会深刻地影响软件的质量和可维护性。为了保证程序编码的质量,程序员必须深刻理解、熟练掌握并正确地运用程序设计语言的特性。此外,还要求源程序具有良好的结构性和良好的程序设计风格。,4.1 编码阶段的任务,模块的过程性描述(不可执行的),源程序(可执行的),二.源程序要求正确可靠、简明清晰、效率高。1.源程序的正确性是对程序质量的最基本要求;2.源程序的简明清晰,才便于验证源代码和模块规格说明的一致 性,容易进行测试和维护;3.源程序的清晰

2、与效率之间常存在矛盾,要求清晰性好的程序一 般效率较低,而要求效率高的程序一般清晰性较差。对于大多 数模块,编码时应该把简明清晰放在第一位。4.除了编码阶段产生源代码外,在测试阶段也需要编写一些测试 程序,用于对软件的测试。,一.编码的目的:使用选定的程序设计语言,把模块的过程性描述翻 译为用语言书写的源程序(源代码)。,补充:程序设计语言的发展(1),面向机器的语言,二.面向机器的语言 包括第一代机器语言和第二代汇编语言。这两种语言都依赖于计算机结构,其指令系统随机器而异。为它所支持的计算机独有。该语言编程效率低、容易出错,且维护困难。,一.程序设计语言的发展 按软件工程的观点,语言的发展大

3、致可分为四代、三个阶段。如图:,ALGOL语言随着Pascal语言的问世,也被新语言取代。,用于一般数值计算与事务处理。简单易懂、具有交互性,是初学者学习程序设计语言的入门语言。,早期版本不支持结构程序设计,解释执行速度慢,不适用于大的软件开发。新版本有较大改进。,主要用于科学计算。新版本在支持结构化控制和字符串处理方面有较大改进。,数据类型不丰富,对复杂的数据结构缺乏支持。,商业数据处理。广泛支持与事务数据处理有关的各种过程技术。近似自然语言,易于理解。,计算功能弱,编译速度不快。,三.高级语言:不依赖于实现它的计算机,它使用的概念和符号接近于自然的 语言。按其特点分为三类:1.基础语言:如

4、BASIC、FORTRAN、COBOL、ALGOL等。该类语言历史久、流传广,且有用它们开发的大量软件,它们的版本 不断进行改进,现今仍被广泛的使用。,科学计算、数据处理及系统软件开发。模块清晰、控制结构完备,有丰富的数据类型和数据结构,表达能力强,易于移植。但不支持大型软件的开发和设计,有希望成为微机和小型机上的软件开发语言。,具有结构化语言的公共特征,表达简洁、控制结构与数据结构完备,移植力强、编译质量高。不仅能写出效率高的应用软件,也适用于编写OS、编译程序等系统软件。如UNIX中90%以上的代码是C语言。,面向过程的现代语言,适用于一切“嵌入式计算机系统”除具有许多经典语言的特征外,还

5、包含许多新特征。支持并发处理与过程间通信,支持异常处理中实施中断及只能由汇编语言实现的低级操作等。在语言的表达与结构上又具有高级语言特点,远比汇编语言易于开发和维护。它是第一个充分体现软件工程思想的语言,它既是编码语言,又可用作设计表达工具。,2.现代语言:如PASCAL、C、Ada等;它们都支持结构化的控制结构,具有很强的过程结构和数据结构能力,因此也称为结构化语言(structured language)。,3.专用语言 基础语言与现代语言都属于通用语言。专用语言不同,它是 为某一特定的应用专门设计的语言。例如:(1)专用于处理数组和向量的APL语言;(2)数控机床用的APT语言;(3)适

6、用于微处理器开发的FORTH语言;(4)适用于处理符号和表达式的LISP语言等。从软件工程的角度看,这类语言仅支持特殊的应用,把软件需 求转换为设计,再从设计转换为实现,较通用语言容易。但它们 的可移植性与可维护性都不及通用语言。,过程化语言。,非过程化语言。,面向过程,说明“怎么做”。,面向应用,说明“做什么”。,高效的实现各种算法。,直接地实现各类应用系统。,着重算法描述,一次仅处理一个记录或数据元素。,以数据或知识为基础,以对集合的处理代替对于单元记录或元素的处理。,不支持对大量共享数据的处理。,支持对大型数据库的高效处理。,四.第四代语言 1.第三代语言(3GL)与第四代语言(4GL)

7、的区别。,2.第四代语言(4GL)应具备的特征(1)具有很强的数据管理能力,能对数据库进行有效的存取、查 询和其它有关操作;(2)能提供一组高效的、非过程化的命令,组成语言的基本语句。编程时用户只需用这些命令说明“做什么”,不必描述实现的 细节;(3)能满足多功能、一体化的要求。为此,语言中除必须含有控 制程序逻辑与实现数据库操作的语句外,还应包括生成与处 理报表、表格、图形,以及实现数据运算和分析统计功能的 各种语句,共同构成一个一体化的语言,以适应多种应用开 发的需要。例如,SQL关系数据语言是最早的4GL语言。,补充:程序设计语言的特点和选择,一.程序设计语言的特点 1.名字说明:程序中

8、使用对象的名字,能为编译程序所检查和识别;2.类型说明:定义对象的类型,确定该对象的使用方式;3.初始化:为变量提供适当的初始值或由系统给变量赋一特殊的表明 尚未初始化的值;4.程序对象的局部性:程序中真正需要的那部分才能访问的对象;5.程序模块:控制程序对象的名字;6.循环控制结构:如FOR语句、WHILE-DO语句、REPEAT-UNTIL语句等;7.分支控制结构:如IF语句、CASE语句等;8.异常处理:为程序运行过程中发生的错误和意外事件提供检测和处 理上的帮助;9.独立编译:能分别编译各个程序单元。,程序设计语言的特点和选择,二.程序设计语言的选择 1.为用户熟悉、便于用户维护的语言

9、。2.运行目标系统的环境中可以提供的编译程序所能选用的语言。3.根据可以得到的软件工具,能在支持程序开发中可以利用的语言。4.根据工程规模的大小、目标系统应用范围选择专用的语言,如实时 应用选择Ada语言或汇编语言,系统软件开发选择C语言或汇编语言,软件开发中若含有大量数据操作则选择SQL、dBASE、VF等数据库语言等。5.程序员熟悉的语言。6.标准化程度高、程序可移植性好的语言。7.根据算法与计算的复杂性、数据结构的复杂性。如对于系统程序和 结构复杂的应用程序,选择支持数组、记录(或结构)与指针的动态 数据结构的Pascal语言或C语言。8.根据实时要求系统需要的响应速度和效率。,4.2

10、结构化程序设计,结构化程序设计主要包括两方面:在编写程序时,强调使用几种基本控制结构,通过组合嵌套,形成程序的控制结构。尽可能避免使用GOTO语句。在程序设计过程中,尽量采用自顶向下和逐步细化的原则,由粗到细,一步步展开。,4.2.1结构化程序设计的主要原则,使用语言中的顺序、选择、重复等有限的基本控制结构表示程序逻辑。选用的控制结构只准许有一个入口和一个出口。程序语句组成容易识别的块,每块只有一个入口和一个出口。复杂结构应该用基本控制结构进行组合嵌套来实现。,语言中没有的控制结构,可用一段等价的程序段模拟,但要求该程序段在整个系统中应前后一致。严格控制GOTO语句,仅在下列情形才可使用:用一

11、个非结构化的程序设计语言去实现一个结构化的构造。若不使用GOTO语句就会使程序功能模糊。在某种可以改善而不是损害程序可读性的情况下。,例1 打印A,B,C三数中最小者程序,程序1 if(A B)goto 120;if(B C)goto 110;100 write(C);goto 140;110 write(B);goto 140;120 if(A C)goto 130;goto 100;130 write(A);140 end,程序2 if(A B)and(A C)then write(A)else if(A B)and(B C)then write(B)else write(C)endif

12、endif,(略)例2 用二分法求方程 f(x)0 在区间a.b中的根的程序,假设在闭区间a.b上函数 f(x)有唯一的一个零点,f0=f(a);f1=f(b);/程序1 if(f0*f1 0)x0=xm;f0=fm;else x1=xm;,finish:printf(“n The root of this equation is%dn”,xm);单入口,两出口正常出口是循环达到 n 次,非正常出口是循环中途控制转出到标号 finish 所在位置可读性好,f0=f(a);f1=f(b);/程序2 if(f0*f1=0)x0=a;x1=b;for(i=1;i=n;i+)xm=(x0 x1)/2;

13、fm=f(xm);if(abs(fm)eps|abs(x1x0)eps)break;if(f0*fm0)x0=xm;f0=fm;else x1=xm;,f0=f(a);f1=f(b);/程序3 if(f0*f1=0)x0=a;x1=b;i=1;finished=0;while(i=n&finished=0)xm=(x0 x1)2;fm=f(xm);if(abs(fm)eps|abs(x1x0)eps)finished=1;if(finished=0),if(f0*fm0)x0=xm;f0=fm;else x1=xm;引入布尔变量 finished,改 for 型循环为 while 型,将单入口

14、多出口结构改为单入口单出口结构。,自顶向下,逐步求精,在详细设计和编码阶段,应当采取自顶向下,逐步求精的方法。把一个模块的功能逐步分解,细化为一系列具体的步骤,进而翻译成一系列用某种程序设计语言写成的程序。,例:用筛选法求100以内的素数,筛选法就是从2到100中去掉2,3,9,10的倍数,剩下的就是100以内的素数。为了解决这个问题,可先按程序功能写出一个框架。,main()/程序框架 建立2到100的数组A,其中Aii;-1 建立2到10的素数表 B,其中存放2 到10以内的素数;-2 若Aii是B 中任一数的倍数,则 剔除Ai;-3 输出A 中所有没有被剔除的数;-4,main()/*建

15、立2到100的数组A,其中Aii*/for(i=2;i=100;i+)Ai=i;/*建立2到10的素数表B,其中存放2到 10以内的素数*/B1=2;B2=3;B3=5;B4=7;/*若Aii是B 中任一数的倍数,则剔 除Ai*/for(j=1;j=4;j+)检查A 所有的数能否被Bj整除并将 能被整除的数从A 中剔除;-3.1,/*输出A 中所有没有被剔除的数*/for(i=2;i=100;i+)若Ai没有被剔除,则输出之-4.1对框架中的局部再做细化,得到整个程序。,main()/*建立2到100的数组A,其中Aii*/for(i=2;i=100;i+)Ai=i;/*建立2到10的素数表B

16、,其中存放2到 10以内的素数*/B1=2;B2=3;B3=5;B4=7;/*若Aii是B 中任一数的倍数,则剔除 Ai*/for(j=1;j=4;j+)/*检查A 所有的数能否被Bj整除并将能 被整除的数从A 中剔除*/,for(i=2;i=100;i+)if(AiBj*Bj=Ai)Ai=0;/*输出A 中所有没有被剔除的数*/for(i=2;i=100;i+)/*若Ai没有被剔除,则输出之*/if(Ai!=0)printf(“A%d%dn”,I,Ai);,自顶向下,逐步求精方法的优点,符合人们解决复杂问题的普遍规律。可提高软件开发的成功率和生产率用先全局后局部,先整体后细节,先抽象后具体的

17、逐步求精的过程开发出来的程序具有清晰的层次结构,程序容易阅读和理解,程序自顶向下,逐步细化,分解成一个树形结构。在同一层的节点上的细化工作相互独立。有利于编码、测试和集成每一步工作仅在上层节点的基础上做不多的设计扩展,便于检查有利于设计的分工和组织工作。,4.3 程序设计风格,程序实际上也是一种供人阅读的文章,有一个文章的风格问题。应该使程序具有良好的风格。源程序文档化 数据说明 语句结构 输入输出方法,一.源程序文档化,符号名的命名程序注释 程序的视觉组织,1.符号名的命名,符号名即标识符,包括模块名、变量名、常量名、标号名、子程序名、数据区名以及缓冲区名等。这些名字应能反映它所代表的实际东

18、西,应有一定实际意义。例如,表示次数的量用Times,表示总量的用Total,表示平均值的用Average,表示和的量用Sum等。,名字不是越长越好,应当选择精炼的意义明确的名字。必要时可使用缩写名字,但这时要注意缩写规则要一致,并且要给每一个名字加注释。同时,在一个程序中,一个变量只应用于一种用途。例:NBALAP N,2.程序的注释,夹在程序中的注释是程序员与日后的程序读者之间通信的重要手段。注释决不是可有可无的。一些正规的程序文本中,注释行的数量占到整个源程序的13到12,甚至更多。注释分为序言性注释和功能性注释。,序言性注释,通常置于每个程序模块的开头部分,它应当给出程序的整体说明,对

19、于理解程序本身具有引导作用。有些软件开发部门对序言性注释做了明确而严格的规定,要求程序编制者逐项列出。有关项目包括:程序标题;,有关本模块功能和目的的说明;主要算法;接口说明:包括调用形式,参数描述,子程序清单;有关数据描述:重要的变量及其用途,约束或限制条件,以及其它有关信息;模块位置:在哪一个源文件中,或隶属于哪一个软件包;开发简历:模块设计者,复审者,复审日期,修改日期及有关说明等。,功能性注释,功能性注释嵌在源程序体中,用以描述其后的语句或程序段是在做什么工作,或是执行了下面的语句会怎么样。而不要解释下面怎么做。例如,/*ADD AMOUNT TO TOTAL*/TOTAL=AMOUN

20、TTOTAL不好。,如果注明把月销售额计入年度总额,便使读者理解了下面语句的意图:/*ADD MONTHLY-SALES TO ANNUAL-TOTAL*/TOTAL=AMOUNTTOTAL要点 描述一段程序,而不是每一个语句;用缩进和空行,使程序与注释容易区别;注释要正确。,3.视觉组织 空格、空行和移行,恰当地利用空格,可以突出运算的优先性,避免发生运算的错误。例如,将表达式(A17)ANDNOT(B49)ORC写成(A17)AND NOT(B49)OR C自然的程序段之间可用空行隔开;,移行也叫做向右缩格。它是指程序中的各行不必都在左端对齐,都从第一格起排列。这样做使程序完全分不清层次关

21、系。对于选择语句和循环语句,把其中的程序段语句向右做阶梯式移行。使程序的逻辑结构更加清晰。例如,两重选择结构嵌套,写成下面的移行形式,层次就清楚得多。,IF()THEN IF()THEN ELSE ENDIF ELSE ENDIF,二、数据说明,在设计阶段已经确定了数据结构的组织及其复杂性。在编写程序时,则需要注意数据说明的风格。为了使程序中数据说明更易于理解和维护,必须注意以下几点。1.数据说明的次序应当规范化 2.说明语句中变量安排有序化 3.使用注释说明复杂数据结构,数据说明的次序应当规范化,数据说明次序规范化,使数据属性容易查找,也有利于测试,排错和维护。原则上,数据说明的次序与语法无

22、关,其次序是任意的。但出于阅读、理解和维护的需要,最好使其规范化,使说明的先后次序固定。,例如,在FORTRAN程序中数据说明次序 常量说明 简单变量类型说明 数组说明 公用数据块说明 所有的文件说明在类型说明中还可进一步要求。例如,可按如下顺序排列:整型量说明 实型量说明 字符量说明 逻辑量说明,说明语句中变量安排有序化,当多个变量名在一个说明语句中说明时,应当对这些变量按字母的顺序排列。带标号的全程数据(如FORTRAN的公用块)也应当按字母的顺序排列。例如,把 integer size,length,width,cost,price写成 integer cost,length,price

23、,size,width,使用注释说明复杂数据结构,如果设计了一个复杂的数据结构,应当使用注释来说明在程序实现时这个数据结构的固有特点。例如,对PL/1的链表结构和Pascal中用户自定义的数据类型,都应当在注释中做必要的补充说明。,三、语句结构,在设计阶段确定了软件的逻辑流结构,但构造单个语句则是编码阶段的任务。语句构造力求简单,直接,不能为了片面追求效率而使语句复杂化。,1.在一行内只写一条语句,在一行内只写一条语句,并且采取适当的移行格式,使程序的逻辑和功能变得更加明确。许多程序设计语言允许在一行内写多个语句。但这种方式会使程序可读性变差。因而不可取。,例如,有一段排序程序FOR I:=1

24、 TO N1 DO BEGIN T:=I;FOR J:=I1 TO N DO IF AJAT THEN T:=J;IF TI THEN BEGIN WORK:=AT;AT:=AI;AI:=WORK;END END;由于一行中包括了多个语句,掩盖了程序的循环结构和条件结构,使其可读性变得很差。,FOR I:=1 TO N-1 DO/改进布局 BEGIN T:=I;FOR J:=I1 TO N DO IF AJAT THEN T:=J;IF TI THEN BEGIN WORK:=AT;AT:=AI;AI:=WORK;END END;,2.程序编写首先应当考虑清晰性,程序编写首先应当考虑清晰性,不

25、要刻意追求技巧性,使程序编写得过于紧凑。例如,有一个用 C 语句写出的程序段:AI=AIAT;AT=AIAT;AI=AIAT;,此段程序可能不易看懂,有时还需用实际数据试验一下。实际上,这段程序的功能就是交换AI和AT中的内容。目的是为了节省一个工作单元。如果改一下:WORK=AT;AT=AI;AI=WORK;就能让读者一目了然了。,3.程序要能直截了当地说明程序员的用意。,程序编写得要简单,写清楚,直截了当地说明程序员的用意。例如,for(i=1;i=n;i+)for(j=1;j=n;j+)Vij(ij)*(ji)除法运算()在除数和被除数都是整型量时,其结果只取整数部分,而得到整型量。,当

26、 ij 时,i/j=0 当 ji 时,j/i=0 得到的数组 当ij时 Vij=(ij)*(ji)=0 当ij时 Vij=(ij)*(ji)=1这样得到的结果 V 是一个单位矩阵。,写成以下的形式,就能让读者直接了解程序编写者的意图。for(i1;i=n;i+)for(j1;j=n;j+)if(i=j)Vij 1.0;ELSE Vij 0.0;,4.除非对效率有特殊的要求,程序编写要做到清晰第一,效率第二。不要为了追求效率而丧失了清晰性。事实上,程序效率的提高主要应通过选择高效的算法来实现。5.首先要保证程序正确,然后才要求提高速度。反过来说,在使程序高速运行时,首先要保证它是正确的。,6.避

27、免使用临时变量而使可读性下降。例如,有的程序员为了追求效率,往往喜欢把表达式 AI1AI;写成 AIAI;XAI1AI;这样将一句分成两句写,会产生意想不到的问题。,7.让编译程序做简单的优化。8.尽可能使用库函数9.避免不必要的转移。同时如果能保持程序可读性,则不必用 GO TO语句。例如,有一个求三个数中最小值的程序:,IF(X Y)GOTO 30 IF(Y Z)GOTO 50 SMALLZ GOTO 7030 IF(X Z)GOTO 60 SMALLZ GOTO 7050 SMALLY GOTO 7060 SMALLX70 CONTINUE,程序只需编写成:smallx;if(y sma

28、ll)smally;if(z small)smallz;所以程序应当简单,不必过于深奥,避免使用GOTO语句绕来绕去。10.尽量只采用三种基本的控制结构来编写程序。除顺序结构外,使用 if-then-else来实现选择结构;使用 do-until或do-while来实现循环结构。,11.避免使用空的ELSE语句和IF THEN IF的语句。这种结构容 易使读者产生误解。例如,if(char=a)if(char=z)cout“This is a letter。”;else cout“This is not a letter。”;可能产生二义性问题。,12.避免采用过于复杂的条件测试。13.尽量减

29、少使用“否定”条件的条件语句。例如,如果在程序中出现 if(!(char0|char 9)改成 if(char=0&char=9)不要让读者绕弯子想。,14.尽可能用通俗易懂的伪码来描述程序的流程,然后再翻译成必须使用的语言。15.数据结构要有利于程序的简化。16.要模块化,使模块功能尽可能单一化,模块间的耦合能够清晰可见。17.利用信息隐蔽,确保每一个模块的独立性。,18.从数据出发去构造程序。19.不要修补不好的程序,要重新编写。也不要一味地追求代码的复用,要重新组织。20.对太大的程序,要分块编写、测试,然后再集成。21.对递归定义的数据结构尽量使用递归过程。,四、输入和输出,输入和输出

30、信息是与用户的使用直接相关的。输入和输出的方式和格式应当尽可能方便用户的使用。一定要避免因设计不当给用户带来的麻烦。因此,在软件需求分析阶段和设计阶段,就应基本确定输入和输出的风格。系统能否被用户接受,有时就取决于输入和输出的风格。,不论是批处理的输入输出方式,还是交互式的输入输出方式,在设计和编码时都应考虑下列原则:1.对所有的输入数据都要进行检验,识别错误的输入,以保证每个数据的有效性;2.检查输入项的各种重要组合的合理性,必要时报告输入状态信息;3.使得输入的步骤和操作尽可能简单,并保持简单的输入格式;,4.输入数据时,应允许使用自由格式输入;5.应允许缺省值;6.输入一批数据时,最好使

31、用输入结束标志,而不要由用户指定输入数据数目;7.在交互式输入输入时,要在屏幕上使用提示符明确提示交互输入的请求,指明可使用选择项的种类和取值范围。同时,在数据输入的过程中和输入结束时,也要在屏幕上给出状态信息;,8.当程序设计语言对输入输出格式有严格要求时,应保持输入格式与输入语句的要求的一致性;9.给所有的输出加注解,并设计输出报表格式。输入输出风格还受到许多其它因素的影响。如输入输出设备(例如终端的类型,图形设备,数字化转换设备等)、用户的熟练程度、以及通信环境等。,4.4 程序效率,讨论效率的准则程序的效率是指程序的执行速度及程序所需占用的内存的存储空间。程序编码是最后提高运行速度和节

32、省存储的机会,因此在此阶段不能不考虑程序的效率。让我们首先明确讨论程序效率的几条准则,效率是一个性能要求,应当在需求分析阶段给出。软件效率以需求为准,不应以人力所及为准。好的设计可以提高效率。程序的效率与程序的简单性相关。一般说来,任何对效率无重要改善,且对程序的简单性、可读性和正确性不利的程序设计方法都是不可取的。,算法对效率的影响,源程序的效率与详细设计阶段确定的算法的效率直接有关。在详细设计翻译转换成源程序代码后,算法效率反映为程序的执行速度和存储容量的要求。设计向程序转换过程中的指导原则:,在编程序前,尽可能化简有关的算术表达式和逻辑表达式;仔细检查算法中的嵌套的循环,尽可能将某些语句

33、或表达式移到循环外面;尽量避免使用多维数组;尽量避免使用指针和复杂的表;采用“快速”的算术运算;,不要混淆数据类型,避免在表达式中出现类型混杂;尽量采用整数算术表达式和布尔表达式;选用等效的高效率算法;许多编译程序具有“优化”功能,可以自动生成高效率的目标代码。,影响存储器效率的因素,在大中型计算机系统中,存储限制不再是主要问题。在这种环境下,对内存采取基于操作系统的分页功能的虚拟存储管理。存储效率与操作系统的分页功能直接有关。,采用结构化程序设计,将程序功能合理分块,使每个模块或一组密切相关模块的程序体积大小与每页的容量相匹配,可减少页面调度,减少内外存交换,提高存储效率。,在微型计算机系统

34、中,存储器的容量对软件设计和编码的制约很大。因此要选择可生成较短目标代码且存储压缩性能优良的编译程序,有时需采用汇编程序。提高存储器效率的关键是程序的简单性。,影响输入输出的因素,输入输出可分为两种类型:面向人(操作员)的输入输出 面向设备的输入输出如果操作员能够十分方便、简单地录入输入数据,或者能够十分直观、一目了然地了解输出信息,则可以说面向人的输入输出是高效的。,关于面向设备的输入/输出,可以提出一些提高输入/输出效率的指导原则:输入/输出的请求应当最小化;对于所有的输入/输出操作,安排适当的缓冲区,以减少频繁的信息交换。对辅助存储(例如磁盘),选择尽可能简单的,可接受的存取方法;,对辅

35、助存储的输入/输出,应当成块传送;对终端或打印机的输入/输出,应考虑设备特性,尽可能改善输入/输出的质量和速度;任何不易理解的,对改善输入/输出效果关系不大的措施都是不可取的;任何不易理解的所谓“超高效”的输入/输出是毫无价值的;,4.4 程序复杂性度量,程序复杂性主要指模块内程序的复杂性。它直接关联到软件开发费用的多少,开发周期的长短和软件内部潜伏错误的多少。减少程序复杂性,可提高软件的简单性和可理解性,并使软件开发费用减少,开发周期缩短,软件内部潜藏错误减少。,复杂性度量需要满足的假设,为了度量程序复杂性,要求:它可以用来计算任何一个程序的复杂性;对于不合理的程序,例如对于长度动态增长的程

36、序,或者对于原则上无法排错的程序,不应当使用它进行复杂性计算;如果程序中指令条数、附加存储量、计算时间增多,不会减少程序的复杂性。,1.代码行度量法,源代码行数度量法基于两个前提:程序复杂性随着程序规模的增加不均衡地增长;控制程序规模的方法最好是采用分而治之的办法。将一个大程序分解成若干个简单的可理解的程序段。,方法的基本考虑是统计一个程序模块的源代码行数目,并以源代码行数做为程序复杂性的度量。设每行代码的出错率为每100行源程序中可能有的错误数目。Thayer曾指出,程序出错率的估算范围是从0.047之间,即每100行源程序中可能存在0.047个错误。他还指出,每行代码的出错率与源程序行数之

37、间不存在简单的线性关系。,Lipow指出,对于小程序,每行代码出错率为1.31.8;对于大程序,每行代码的出错率增加到2.73.2之间,这只是考虑了程序的可执行部分,没有包括程序中的说明部分。Lipow及其他研究者得出一个结论:对于少于100个语句的小程序,源代码行数与出错率是线性相关的。随着程序的增大,出错率以非线性方式增长。,2.McCabe度量法,McCabe度量法,又称环路复杂性度量,是一种基于程序控制流的复杂性度量方法。它基于一个程序模块的程序图中环路的个数,因此计算它先要画出程序图。程序图是退化的程序流程图。流程图中每个处理都退化成一个结点,流线变成连接不同结点的有向弧。,程序图仅

38、描述程序内部的控制流程,完全不表现对数据的具体操作,以及分支和循环的具体条件。计算环路复杂性的方法:根据图论,在一个强连通的有向图G中,环的个数由以下公式给出:V(G)mnp其中,V(G)是有向图G中环路个数,m是图G中弧数,n是图G中结点数,p是图G中的强连通分量个数。,Myers建议,对于复合判定,例如,(A0)(CD)(XA)算做三个判定。为使图成为强连通图,从图的入口点到出口点加一条用虚线表示的有向边,使图成为强连通图。这样就可以使用上式计算环路复杂性。在例示中,结点数n11,弧数m13,p1,则有 V(G)mnp131113.等于程序图中弧所封闭的区域数。,练习:计算下图环路复杂性,

39、图1,图2,练习:计算下图环路复杂性,图3,几点说明,环路复杂度取决于程序控制结构的复杂度。当程序的分支数目或循环数目增加时其复杂度也增加。环路复杂度与程序中覆盖的路径条数有关。环路复杂度是可加的。例如,模块A的复杂度为3,模块B的复杂度为 4,则模块A与模块B的复杂度是7。,McCabe建议,对于复杂度超过10的程序,应分成几个小程序,以减少程序中的错误。Walsh用实例证实了这个建议的正确性。在McCabe复杂度为10的附近,存在出错率的间断跃变。McCabe环路复杂度隐含的前提是:错误与程序的判定加上例行子程序的调用数目成正比。加工复杂性、数据结构、录入与打乱输入卡片的错误可以忽略不计。

40、,这种度量的缺点是:对于不同种类的控制流的复杂性不能区分 简单IF语句与循环语句的复杂性同等看待 嵌套IF语句与简单CASE语句的复杂性是一样的 模块间接口当成一个简单分支一样处理 一个具有1000行的顺序程序与一行语句的复杂性相同,Halstead的软件科学(了解),Halstead软件科学研究确定计算机软件开发中的一些定量规律,它采用以下一组基本的度量值。这些度量值通常在程序产生之后得出,或者在设计完成之后估算出。,程序长度(预测的Halstead长度)令n1表示程序中不同运算符(包括保留字)的个数,令n2表示程序中不同运算对象的个数,令H表示“程序长度”,则有 H=n1log2 n1+n

41、2 log2n2这里,H是程序长度的预测值,它不等于程序中语句个数。,在定义中,运算符包括:算术运算符 赋值符(=或:=)逻辑运算符 分界符(,或;或:)关系运算符 括号运算符 子程序调用符 数组操作符 循环操作符等。特别地,成对的运算符,例如“beginend”、“forto”、“repeat until”、“whiledo”、“ifthenelse”、“()”等都当 做单一运算符。,运算对象包括变量名和常数。实际的Halstead长度设N1为程序中实际出现的运算符总个数,N2为程序中实际出现的运算对象总个数,N为实际的Halstead长度,则有 N=N1+N2,程序的词汇表Halstead

42、定义程序的词汇表为不同的运算符种类数n1和不同的运算对象种类数n2的总和。若令n为程序的词汇表,则有 n=n1+n2例如,用FORTRAN语言写出的交换排序的例子 SUBROUTINE SORT(X,N)DIMENSION X(N),IF(N.LT.2)RETURN DO 20 I=2,N DO 10 J=1,I IF(X(I).GE.X(J)GO TO 10 SAVE=X(I)X(I)=X(J)X(J)=SAVE10 CONTINUE20 CONTINUE RETURN END,程序量 程序量 V 可用下式得到 V=N log2n 它表明了程序在词汇上的复杂性。其最小值为 V*=(2+n2*

43、)log2(2+n2*)V 这里,2表明程序中至少有两个运算符:赋值符=和函数调用符 f(),n2*表示输入输出变量个数。,对于上面的例子,利用n1,N1,n2,N2,可以计算得 H=10 log210+7 log27=52.87 N=28+22=50 V=(28+22)log2(10+7)=204等效的汇编语言程序的V328。这说明汇编语言比FORTRAN语言需要更多的信息量(以bit表示)。,程序量比率(语言的抽象级别)L=V*/V 或 L=(2/n1)(n2/N2)它表明了一个程序的最紧凑形式的程序量与实际程序量之比,反映了程序的效率。其倒数D=1/L 表明了实现算法的困难程度。,程序员

44、工作量 E=V/L 程序的潜在错误Halstead度量可以用来预测程序中的错误。预测公式为 B=(N1+N2)log2(n1+n2)/3000 B为该程序的错误数。它表明程序中可能存在的差错 B 应与程序量V成正比。,例如,一个程序对75个数据库项共访问1300次,对150个运算符共使用了1200次,那么预测该程序的错误数:B=(1200+1300)log2(75+150)/3000 6.5 即预测该程序中可能包含67个错误,Halstead的重要结论程序的实际Halstead长度N可以由词汇表n算出。即使程序还未编制完成,也能预先算出程序的实际Halstead长度N,虽然它没有明确指出程序中

45、到底有多少个语句。这个结论非常有用。经过多次验证,预测的Halstead长度与实际的Halstead长度是非常接近的。,Halstead度量的缺点,没有区别自己编的程序与别人编的程序。这是与实际经验相违背的。这时应将外部调用乘上一个大于1的的常数Kf(应在15之间,它与文档资料的清晰度有关)。没有考虑非执行语句。补救办法:在统计n1、n2、N1、N2时,可以把非执行语句中出现的运算对象,运算符统计在内。,在允许混合运算的语言中,每种运算符与它的运算对象相关。如果一种语言有整型、实型、双精度型三种不同类型的运算对象,则任何一种基本算术运算符(、)实际上代表了=6 种运算符。在计算时应考虑这种因数

46、据类型而引起差异的情况。,没有注意调用的深度。Halstead 公式应当对调用子程序的不同深度区别对待。在计算嵌套调用的运算符和运算对象时,应乘上一个调用深度因子。这样可以增大嵌套调用时的错误预测率。没有把不同类型的运算对象,运算符与不同的错误发生率联系起来,而是把它们同等看待。例如,对简单if语句与while语句就没有区别。,忽视了嵌套结构(嵌套的循环语句、嵌套IF语句、括号结构等)。一般地,运算符的嵌套序列,总比具有相同数量的运算符和运算对象的非嵌套序列要复杂得多。解决的办法是对嵌套结果乘上一个嵌套因子。,小结,1.编码的目的是把详细设计的结果翻译成用选定语言书写的源 程序。2.程序的质量主要是由设计的质量决定的.但是,编码的风格和 使用的语言,对编码质量也有重要的影响。3.良好的编码风格,应该以结构程序设计原则为指导,使用单 入口、单出口的控制结构,有规律地使用GOTO语句。4.提倡源代码文档化。5.程序的输入和输出应充分考虑运行工程学的要求,在满足数 据可靠性的前提下,尽量作到对用户友好。6.了解各种语言的主要特点,以便根据问题的需要,合理的选 择适当的编码语言。7.编码工具对提高编程的速度和效率是不可缺少的。,要求:在课程报告中简单写一写你们在实际编码时的体会(例如:选择哪一种语言、编代码时结构有无影响,出错时的改进等。),

展开阅读全文
相关资源
猜你喜欢
相关搜索
资源标签

当前位置:首页 > 生活休闲 > 在线阅读


备案号:宁ICP备20000045号-2

经营许可证:宁B2-20210002

宁公网安备 64010402000987号