收藏 分享(赏)

第7章 程序设计语言.ppt

上传人:tkhy51908 文档编号:8010309 上传时间:2019-06-04 格式:PPT 页数:94 大小:472KB
下载 相关 举报
第7章 程序设计语言.ppt_第1页
第1页 / 共94页
第7章 程序设计语言.ppt_第2页
第2页 / 共94页
第7章 程序设计语言.ppt_第3页
第3页 / 共94页
第7章 程序设计语言.ppt_第4页
第4页 / 共94页
第7章 程序设计语言.ppt_第5页
第5页 / 共94页
点击查看更多>>
资源描述

1、第7章 程序设计语言,7.1 程序设计语言的发展历史 7.2 高级语言的基本元素 7.3 高级语言的编译,7.1 程序设计语言的发展历史,7.1.1 程序设计语言的断代划分程序设计语言发展到目前,共经历了三代,它们分别是机器语言时代、汇编语言时代和高级语言时代。1. 机器语言计算机刚发明时使用的程序设计语言是机器语言。机器语言的程序由指令组成,机器语言的特点是所有指令的记号都采用符号0、1的编码组成。,我们在第3章中较详细地介绍了机器指令的表示方法,并给出了一个机器语言形式的程序例子。对比机器语言程序的例子和本章将要讨论的高级语言程序的例子,可以发现,机器语言每条指令的功能很弱。因此,用机器语

2、言书写程序时,程序设计人员不仅非常费力,而且编写程序的效率还非常低。另外,不同计算机的机器语言是不相同的,因此,用机器语言书写的程序在不同的计算机上不能通用。这样,当要把一个程序在另外类型的一个计算机系统上运行时,就需要重新编写程序代码。第3章我们曾讨论过一个简单的实现两个整数值相加功能的机器语言程序,其十六进制码格式的机器语言程序如表7-1所示。,表7-1 机器语言程序,2. 汇编语言对程序设计人员来说,用机器语言书写程序的最大困难是把算法转换为由0、1符号编码表示的指令。因此,人们就试图用容易记忆和交流的符号来分别表示机器语言中的操作码和操作数,这就形成了汇编语言。汇编语言出现于20世纪5

3、0年代初期。汇编语言是程序设计语言发展的第二代。,机器语言中用术语机器指令来表示机器语言中某个特殊的操作。类似地,汇编语言中用术语指令来表示汇编语言中某个特殊的操作。汇编语言和机器语言基本上是一一对应的。也就是说,对大多数汇编语言中的指令来说,在机器语言中都存在一条功能相同的机器指令。例如,假设汇编语言中用LOAD表示取数操作,对应机器指令的操作码为10;,用STORE表示存数操作,对应机器指令的操作码为20;用ADD表示加法操作,对应机器指令的操作码为30;用HALT表示结束程序运行操作,对应机器指令的操作码为00。再假设汇编语言中用X表示内存单元地址20,用Y表示内存单元地址21,用SUM

4、表示内存单元地址22,则上述实现两个整数值相加功能的汇编语言程序如表7-2所示。,表7-2 汇编语言程序,显然,和使用机器语言编写程序比较,使用汇编语言编写程序要容易许多。当然,计算机不能直接理解和执行用汇编语言编写的程序,需要进行转换。因为汇编语言的指令和机器语言的机器指令存在对应关系,所以这样的转换并不困难。汇编程序就是完成这种转换工作的一种专门的程序。汇编程序是把用汇编语言编写的程序(术语称为源程序)翻译为等价的机器语言程序(术语称为目标程序)的一种程序。,现在程序设计语言已发展到了第三代的高级语言,对大多数软件设计问题来说,使用高级语言编写程序是一种更为方便的选择。但是,由于在把用高级

5、语言编写的程序翻译为等价的机器语言程序的过程中,不一定能达到计算机运行效率最高,所以,在一些特定的软件设计问题中,仍然使用汇编语言编写程序。例如,目前大多数外部设备的驱动程序都是用汇编语言编写的。,我们在第7章讨论了伪码形式的算法表示。伪码用非常接近自然语言的描述形式来描述算法。高级语言是一种在伪码形式的算法表示基础上的计算机程序设计语言,因此,和汇编语言相比,高级程序设计语言的抽象度高,和具体计算机的相关度低(或没有相关度),求解问题的方法描述直观。由于高级语言的这些特点,所以,用高级语言设计程序的难度较以前大大降低。,最早出现的高级语言是FORTRAN语言。1957年,巴克斯领导的研究小组

6、开发出了FORTRAN语言,其名字是FORmula TRANslator(公式翻译)的缩写。FORTRAN语言允许程序设计人员直接把计算公式书写在程序中,这就大大简化了程序设计的复杂性,提高了程序设计的效率。而把用FORTRAN语言编写的程序转换成机器语言程序的任务由一个称为FORTRAN语言编译程序的程序来完成。FORTRAN语言编译程序的功能在某种意义上类似于汇编程序,只是由于FORTRAN语言和机器语言的对应关系复杂,所以FORTRAN语言编译程序较汇编程序复杂许多。,FORTRAN语言受到了程序设计人员的一致好评。FORTRAN语言的成功为高级语言的发展奠定了基础。随着软件技术的发展,

7、人们对FORTRAN语言不断进行改造和更新。1958年,FORTRAN被修订为FORTRAN ,几年后又被修订为FORTRAN 。不同的FORTRAN语言软件开发商实现了FORTRAN语言的不同版本,这些版本之间互不兼容,这给程序的共享和程序的移植造成了很大的困难。1966年,FORTRAN 以FORTRAN 66的名称成为一种标准。此后,FORTRAN语言的标准又两度更新FORTRAN 77和FORTRAN 90。,随着FORTRAN语言的成功和不断发展,又有许多高级程序设计语言被提出,如ALGOL 语言、COBOL 语言等,在程序设计语言几十年的发展历程中,曾经推出的高级语言不下百种。随着

8、程序设计语言的不断更新和发展,许多高级语言由于先天的不足,或后天没有软件商对其进行持续的更新和改造,所以,许多高级语言逐渐被市场淘汰,如ALGOL 语言目前就不再被使用。,7.1.2 高级语言的分类高级语言有许多种分类方法。常用的高级语言分类方法有:按照设计要求分,按照应用范围分,按照描述问题的方式分,等等。其中,按照描述问题的方式对高级语言进行分类,是最常用的分类方法。按照设计要求分,可把高级语言分成过程式语言和非过程式语言。过程式语言的主要特征是设计者通过给出一列可以顺序执行的运算步骤,来描述相应的求解过程。FORTRAN语言、COBOL 语言、ALGOL 语言等是过程式语言的典型代表。,

9、非过程式语言的定义是相对于过程式语言来说的,凡是设计者无法表示出求解过程的一列可以顺序执行的运算步骤的语言都是非过程式语言。非过程式语言的典型代表有 PROLOG 语言、GPSS语言等。例如,用 PROLOG 语言编写的程序是以逻辑推理为问题求解的基础,而不是通过给出一列可以顺序执行的运算步骤来描述求解步骤的。PROLOG 语言程序的执行过程是按照程序语句的逻辑次序来执行的,这种逻辑次序和FORTRAN语言描述的执行过程是完全不相同的。,按照应用范围分,可把高级语言分成通用语言和专用语言。目标专一的语言称为专用语言,目标非专一的语言称为通用语言。专用语言有APT语言等。绝大部分高级语言都属于通

10、用语言。按照描述问题的方式分,可把高级语言分成命令型语言、函数型语言、描述型语言和面向对象语言。命令型语言是出现最早和曾经使用最多的高级语言。命令型语言的特点是计算机按照该语言描述的操作步骤来执行。换句话说,命令型语言程序中的语句就是要求计算机执行的“命令”。FORTRAN语言、COBOL 语言、ALGOL语言、BASIC语言、C 语言、PASCAL语言、Ada 语言、APL 语言等都属于命令型语言。,函数型语言的特点是把问题求解过程表示成“块”结构,对调用“块”的调用者来说,每个块有输入数据和经过加工处理后的输出数据。这样,每个“块”的功能就像数学家所说的“函数”的功能,所以这种语言称为函数

11、型语言。LISP语言、ML语言等都属于函数型语言。如果说命令型语言强调的是求解问题的步骤是什么,那么,描述型语言强调的就是问题是什么。描述型语言的特点是设计者给出的是问题的描述,计算机根据对问题描述的逻辑进行处理。由于这类高级语言是基于逻辑的,所以也称为逻辑型语言。PROLOG语言、GPSS语言等都属于描述型语言。,程序是要处理现实世界中的问题,如果把现实世界中的事物称作对象,那么每个对象都由一组属性和一组行为组成。如“学生”这个对象的属性就有学号、姓名、年龄、性别等,“学生”这个对象的行为就有注册、登记考试成绩等。在命令型语言等程序设计语言中,一个对象的属性和该对象的行为是分离的,而面向对象

12、语言的基础就是把对象的属性和对象的行为结合为一体进行程序设计。Smalltalk语言、C+语言、Visual Basic语言、Java语言等都属于面向对象语言。面向对象语言是目前最为流行的程序设计语言。上述关于四种高级语言特征和典型语言的讨论总结见表7-3。,表7-3 四种类型高级语言一览表,在上述四种类型的高级语言中,早期是命令型语言占主导地位,通常说的传统程序设计语言主要指的是命令型语言。由于以对象为基础的面向对象的高级语言较传统程序设计语言更符合人类思维和求解问题的方式,所以近10年来,面向对象的高级语言有了长足的发展。面向对象的高级语言是目前程序设计语言发展的主流方向。,7.2 高级语

13、言的基本元素,在四种类型的高级语言中,命令型语言和面向对象语言在基本元素的表示和实现方法上基本一致,而函数型语言和描述型语言,在基本元素的表示和实现方法上也和其他两种类型的高级语言很不相同。考虑到函数型语言和描述型语言使用较少,本节关于高级语言基本元素的讨论,主要是针对命令型语言和面向对象语言而言。,6.2节我们讨论了算法的基本元素,这些基本元素是用伪码形式表示的。高级程序设计语言要做到既能方便设计人员编程,又能方便计算机实现(即计算机能方便地最终识别、理解并运行高级语言程序),采用类似于伪码形式的、有固定格式和特定含义的语句作为程序设计的基本元素,就是一个恰当的选择。因此,高级语言的基本元素

14、主要就是算法基本元素的高级语言实现。,7.2.1 变量在算法中,变量的定义是代表某个具体数值,并可改变其数值的符号。在高级语言中,要实现变量,就是要在使用变量前,给每个变量分配所需大小的内存单元空间。因此,高级语言中要求在使用某变量前,首先要定义该变量。从高级语言的角度看,变量代表了特定大小的内存单元空间。高级语言把变量分为全局变量和局部变量。全局变量是允许所有模块都使用的变量,局部变量是只允许在一个过程体内使用的变量。关于全局变量和局部变量的概念我们在7.2.6节讨论过程时再进一步说明。,7.2.2 数据类型由于不同类型的数值占用内存单元的大小不同,所以高级语言在进行变量定义时,要具体指出该

15、变量要存放数值的类型。高级语言中引入数据类型的概念来解决这一问题。在高级语言中,定义变量就是指明该变量的数据类型,从而为该变量分配相应数据类型的内存单元空间。例如,C语言规定,整数有三种数据类型,分别是长整数、整数和短整数,长整数用符号long(或long int)表示,整数用符号int表示,短整数用符号short(或short int)表示。,数据类型占用4个字节,由于4个字节可表示的数据范围是-32 768+32 767,所以,定义为int数据类型变量的数据范围是-32 768+32 767,当定义为int数据类型的变量超出了这个范围,数据就会出错,其程序也就必然会出错。long数据类型占

16、用8个字节,所以,定义为long数据类型的变量可表示的数据范围,就远大于定义为int数据类型的变量可表示的数据范围。,在高级语言中,数据类型通常分作基本数据类型和构造数据类型两种。1. 基本数据类型一般来说,高级语言的数据类型包括整数数据类型、实数数据类型和字符数据类型。为节省内存单元空间,整数数据类型又进一步细分为长整数、整数和短整数三种子类型,实数数据类型又分作单精度和双精度两种子类型。7.2.3节我们还要讨论构造结构的数据类型。由于构造结构的数据类型是由这里讨论的简单数据类型构成的,所以为区别起见,我们也称这里讨论的简单数据类型为基本数据类型。,为有效和正确地进行程序设计,程序设计人员在

17、定义变量时,首先要根据问题分清该变量应是整数、实数和字符数据类型的哪一种,然后还要根据问题分清该变量应是哪种子数据类型。如果把所有变量都按占最大内存单元空间的子数据类型来定义,当然程序运行时不会出错,但是浪费的内存资源很多;如果把所有变量都按占最小内存单元空间的子数据类型来定义,当然内存资源会很节省,但是一旦数据超过该数据类型所允许的数据范围,则程序运行时就会出错。所以,定义变量时,数据类型(特别是子数据类型)的选择要合适。,在高级语言中,数据类型不仅用来确定数据(或变量)所要占用的内存单元大小,还用来进行操作的匹配性检查。例如,假设有如下C语言语句:int n;n = 5.5;上述程序段中,

18、第一行语句定义了一个int类型的变量,第二行语句是给变量n赋一个单精度的实数数值5.5,而一个只有4个字节内存单元的整数类型变量中,是无法存放需要8个字节内存单元空间的单精度实数数值的,因此系统需要在运行该程序前对该语句提出可能存在错误或可能引起错误的警告,并在实际运行时,只把单精度实数数值5.5的整数部分赋值给变量n。,总结上述讨论,我们可以得出如下结论:(1) 数据类型是高级语言定义变量(即为变量分配具体大小的内存单元)的需要。(2) 数据类型是系统用来检查高级语言程序中表达式计算或变量赋值等是否匹配的需要。为下面举例说明方便,这里我们给出C语言中基本数据类型的表示符号:整数:长整数为lo

19、ng(或long int),整数为int,短整数为short(或short int);实数:单精度实数为float,双精度实数为double;字符:char。,2. 构造数据类型现实世界中有很多事物的属性是密切相关的,程序设计中反映这些事物属性的数据也应该密切相关。例如,描述学生的数据有学号、姓名、性别、年龄等,一个学生的这些数据应该组成一个整体。构造数据类型为程序设计人员提供了把现实世界中有密切联系的数据组织成一个整体的工具。构造数据类型是在基本数据类型的基础上构造出来的数据类型。数组和结构是大多数高级语言都支持的两种最主要的构造数据类型。,(1) 数组。数组是相同数据类型的集合。例如,在C

20、语言中,假设我们要使用10个int型的变量,可以定义如下:int v1,v2,v3,v4,v5,v6,v7,v8,v9,v10;为和数组类型的变量区别,我们也称这里定义的变量为简单变量。由于上述10个变量的类型相同,所以,也可以把这10个变量定义成一个数组。不同的高级语言定义数组和表示数组元素的方法略有不同,但基本原理都相同。在定义变量时,C语言用变量名后加符号“ ”来表示所定义的变量为数组类型的变量。如: int v10,上述语句定义了包含10个变量的数组类型变量。数组变量中的一个变量称作一个数组元素。C语言规定数组元素的序号从0开始,所以10个数组元素分别是:v0,v1,v2,v3,v4,

21、v5,v6,v7,v8,v9。由于一个数组变量包含了若干个数组元素,所以高级语言在实现时,是给每个数组变量分配内存地址连续的内存单元块。如数组变量v的内存单元分配示意图如图7-1所示。每个数组元素占4个字节,所以总计占用了40个字节。,图7-1 数组变量的内存单元分配,由于一个数组变量是由若干个数组元素组成的一个整体,所以数组类型既可以简化变量的定义,还可以方便程序设计。例如,如果上述10个数组元素的数组变量中保存了要累加的数值,在编写循环形式的累加时,就可以把循环体写成sum = sum + vi,其中,数组下标i从0增加到9。,(2) 结构体。结构体是不同数据类型的集合。结构体用来表示有若

22、干个分量的一个事物。例如,在登记一个学生的信息时,需要登记学生的姓名、年龄、平均成绩等信息。我们希望把描述一个学生各个属性的信息表示成一个有逻辑联系的整体,这样既可以简化变量的定义,也可以使程序更容易看懂。结构体就可以达到这个目的。,不同的高级语言表示结构体的方法不同。在C语言中,上述学生结构体可以定义如下:struct studentchar name8;int age;float average;,上述语句称作结构体定义,结构体定义是指根据问题的需要,利用基本数据类型定义一种新的数据类型。其中,student为这个结构体的名,name、age和average是该结构体的三个分量。在上述定义

23、之后,就可以像用数据类型int定义变量一样,用结构体student来定义结构体变量。例如,结构体变量可以定义如下:student s;由于一个结构体变量包含了若干个分量,所以高级语言在实现时,是给每个结构体变量分配内存地址连续的内存单元块。如结构体变量s的内存单元分配示意图如图7-2所示。,图7-2 结构体变量的内存单元分配,在上述结构体变量s定义之后,我们就可以用如下三个赋值语句给该变量的三个分量分别赋值:s.name = 张三;s.age = 22;s.average = 89.9;,7.2.3 赋值语句变量代表了特定大小的内存单元空间。在定义变量时,只是给变量分配了所要大小的内存单元空间

24、,这个内存单元中并没有存放任何数值。赋值语句用来实现给变量赋数值。为和人们的使用习惯一致,大多数高级语言的赋值语句都用赋值号“=”表示。赋值号“=”右边是要赋予的数值,称为右值。赋值号“=”左边是要接受数值的变量,称为左值。变量中保存的数值可以用赋值语句修改。例如,在C语言中,下面程序段是允许的:,int n;n = 22;n = n+1;其中,第一条语句定义变量n为int类型的变量,第二条语句给变量n赋予数值22,第三条语句修改变量n中的数值为23。对上述赋值语句还要说明的是,变量出现在左值位置和出现在右值位置表示不同的含义。出现在左值位置的变量是要接受赋值数据的变量,所以,出现在左值位置的

25、变量表示该变量对应的内存单元地址。出现在右值位置的变量是要赋值的数据,所以,出现在右值位置的变量表示该变量对应内存单元中的数值。,7.2.4 分支语句分支语句用来实现多个分支情况下的选择。最基本的分支语句是IF-THEN-ELSE(或其变种IF-THEN)语句。不同的高级语言表示IF语句的格式有所不同。C语言表示IF-THEN-ELSE(或其变种IF-THEN)语句的格式为:if(condition) S1;else S2;C语言规定一条语句结束时用分号“;”标识,这样C语言就把条件语句分成了两条语句。,果只有第一条语句,则表示当条件condition为真时,执行语句组S1;如果两条语句都有,

26、则表示当条件condition为真时,执行语句组S1,否则执行语句组S2。这样就方便地实现了IF-THEN-ELSE语句及其变种IF-THEN语句的功能。当语句组S1或S2为若干条语句时,需要明确地表示出语句组S1或语句组S2包含的语句。C语言用一对花括号“”括起来的部分表示对应的语句组。例如,下列语句:,if(n100)x = x + n;n = n + 2; elsex = x - n;n = n - 2; ,上面语句表示当n100时,执行语句x = x + n和语句n = n + 2;否则执行语句x = x - n和语句n = n - 2。有些高级语言(如PASCAL语言),用标识符“b

27、egin”代表语句组的开始,用标识符“end”代表语句组的结束。当分支多于2个时,虽然也可以用分支语句嵌套来实现,但是毕竟比较麻烦。特别是当分支有3个以上时,这样的表示形式显得很繁琐。对于这种多分支的情况,大多数高级语言提供一种称为CASE的语句,可以方便地实现多分支情况下的程序设计。在C语言中,CASE语句的格式为:,switch(N) case C1: S1;case C2: S2;.case Cn: Sn;default: S0; ,7.2.5 循环语句循环通常有两种情况:一种情况是循环次数不固定,需要根据当前的条件判断来决定是否继续执行循环体;另一种情况是循环次数固定,执行循环体的次数

28、可以明确给出。大部分高级语言都给出了可以方便表示这两种情况的循环语句。例如,C语言对第一种情况的循环语句格式为:while(condition) S该语句的语义是:当条件condition为真时,执行语句组S;当条件condition为假时,执行该语句后面的语句。,C语言对第二种情况的循环语句格式为:For(count = n1; count = n2; count = count + c) S该语句的语义是:初始时,计数变量count 等于数值n1;当计数变量count小于数值n2时,首先执行语句组S,然后把计数变量count修改为count 加上某个常数c;然后继续测试计数变量count是

29、否小于数值n2。这样循环体的语句组S将总共被执行(n2-n1+1)/c次。用第7章讨论的计算1+2+3+100问题的C语言程序来说明两种方式的循环语句。,种循环语句的C语言程序段如下:int sum, n;sum = 0;n = 1; while(n = 100) sum = sum +n; n = n + 1; ,使用第二种循环语句的C语言程序段如下:int sum, n;sum = 0;for(n = 1; n = 100; n = n + 1) sum = sum + n; 上述程序段的执行流程如图7-3所示。,图7-3 for循环执行流程示例,7.2.6 过程我们在7.2节中曾指出,过

30、程是算法的基本元素,过程可以使一个求解大问题的算法分解为若干个求解子问题算法的有机合成。高级语言在实现过程时,要考虑过程的调用、过程的参数、过程的返回、参数的传递方式等问题。1. 过程的调用过程主要有两个用途:(1) 构造通用的算法模块;(2) 把一个大的、复杂的算法分解为若干个小的、简单的算法的合成。,过程的这两种用途都涉及一个程序对一个过程的调用。所谓过程的调用,是指停止执行当前的程序,而转去执行被调用的过程。过程的调用如图7-4所示。在图7-4中,主程序表示当前正在运行的程序。在主程序的运行过程中,如果遇到调用过程的语句,则停止运行当前的主程序,而转去运行被调用的过程。过程运行完后,再返

31、回到主程序调用过程语句的下一条语句继续运行。,由图7-4所示的过程的调用可知,高级语言中必须包含一条过程调用语句。过程调用语句要给出被调用的过程名。不同高级语言的过程调用语句格式不同。下面给出C语言和PASCAL语言的过程调用语句示例,例子中的Sum3表示被调用的过程名,为使问题简单,假设该过程没有参数:C语言 PASCAL语言Sum3; CALL Sum3,图7-4 过程的调用,2过程的参数通常过程都包含有参数,过程的参数使过程的设计具有灵活性和通用性。例如,求1n的累加和问题的C语言函数(C语言把过程称作函数)如下:int Sum1(int n) int i, sum; /变量定义sum

32、= 0; /累加初始化for(i = 1; i = n; i = i + 1),sum = sum + i; /循环累加return sum; /返回语句上述C语言函数可以求1n的累加和问题,其中,n可以是任意的整数数值。这样,高级语言过程的参数就必须是变量,该变量的具体取值由调用该过程的主程序给出。主程序通过传送具体的数值给被调用过程的参数,来实现具体的过程调用。,过程的引入就使程序可以由多个模块组成,这些模块在具体设计时可能由多个不同的人来完成。为了防止一个过程内的变量数值被起了相同变量名字的其他过程混用,高级语言把变量分为全局变量和局部变量。全局变量是允许所有模块都使用的变量,局部变量是

33、只允许在一个过程体内使用的变量。过程的参数是局部变量。例如,上面例子中的参数n是局部变量。,3. 过程的返回从图7-4可知,过程的调用包含调用和返回两个步骤。调用步骤要求主程序传送具体数值给过程,然后通过过程的参数来实现。返回步骤要求把过程的计算结果传送给主程序。由于变量分作局部变量和全局变量,为避免一个过程被外部的程序非法修改,高级语言规定:在一个过程内定义的变量都是局部变量。例如,上述例子中过程内定义的变量sum就是局部变量。我们说过,外部于过程的主程序是无法得到过程内局部变量的数值的,所以要考虑把过程计算结果传送给主程序的方法。,高级语言处理过程返回的第一种方法,是把过程名既看作过程名,

34、也看作一个变量。这样,只要我们首先把过程名这个变量定义成全局变量,然后把过程的计算结果传送给过程名,在主程序中就可以得到计算结果。不同的高级语言实现把过程的计算结果传送给过程名的方法不同。在C语言中,设计了一个专门语句用来实现把过程的计算结果传送给过程名,该语句的格式如下:return x;,4. 参数的传递方式主程序通过给被调用过程的参数传送具体的数值来实现过程的调用。主程序和被调用过程之间通过参数传送数据有两种方法。通过参数传送数据的第一种方法称为值传送。所谓值传送,就是把主程序调用过程的具体数值(称为实际参数,或简称实参)拷贝给过程的参数(称为虚拟参数,或简称虚参)。图7-5(a)就是值

35、传送方式实现方法的示意。其中,实参和虚参都是内存单元,在过程调用时,把实参中的数值拷贝给虚参。,通过参数传送数据的第二种方法称为地址传送。所谓地址传送,就是把虚参表示成内存单元地址形式,此时虚参指向的内存单元和实参指向的内存单元相同,或者说,虚参和实参共享一个内存单元。图7-5(b)就是地址传送方式实现方法的示意。,图7-5 参数的传递方式 (a) 值传送;(b) 地址传送,参数的传递方式对程序实现的方法有一定的影响。对于值传送来说,由于数据的传送是单方向的,即只有从主程序到过程一个方向,所以,如果运行过程时,参数的数值发生了改变,将不会影响到主程序中原来的数值。如图7-6所示,过程调用时,实

36、参拷贝数值6给虚参,若过程运行时虚参的数值改变为8,则过程结束后主程序中实参的数值还是原来的6,而不是8。,图7-6 值传送方式的单向性 (a) 调用时;(b) 运行时;(c) 结束时,对于地址传送来说,由于虚参和实参实际上是同一个内存单元,所以,可以认为数据的传送是双方向的。此时,虚参数值的任何改变都会影响到实参的数值。如图7-7所示,过程调用时,实参的数值为6,则虚参的数值也为6,若过程运行时虚参的数值改变为8,则过程结束后主程序中实参的数值也就是8,而不是原来的6。,图7-7 地址传送方式的双向性 (a) 调用时;(b) 运行时;(c) 结束时,参数的地址传送方式可以把过程的计算结果带回

37、到主程序中。前面我们说过,高级语言处理过程返回的第一种方法,是让定义为全局变量的过程名把过程的计算结果带回到主程序中。有些复杂问题的计算结果有若干个,而过程名却只能带回一个计算结果。参数地址传送方式的双向性,提供了过程返回的第二种方法。我们可以把需要带回主程序的计算结果设计成地址传送方式的虚参,这样,在过程运行完后,主程序就可以从相应的实参得到过程的计算结果。,5. 过程调用示例这里给出一个求1n的累加和问题的完整C+语言程序,以此来说明高级语言过程调用的实现方法。C+语言程序如下:int Sum1(int n) /过程定义int i, sum; sum = 0; for(i = 1; i =

38、 n; i = i + 1)sum = sum + i;,return sum; #include void main(void) /主程序 int n, sum;cout n; /输入nsum = Sum1(n); /过程调用cout “sum = “ sum endl; /输出sum ,过程定义部分既定义了过程Sum1的具体实现,也定义了过程名Sum1是一个全局变量。主程序是以名字为main开始的部分,主程序中首先要求用户输入实参n的具体数值(假设用户输入100),然后以实参n=100调用过程Sum1,过程Sum1完成1n的累加计算后,语句“return sum”实现把局部变量sum中保存

39、的计算结果(数值为5050)传递给过程名Sum1,最后返回主程序继续运行。因为过程名Sum1是一个全局变量,所以主程序中可以读取该变量中的数值。主程序中把全局变量Sum1中保存的计算结果(数值为5050)再赋值给主程序中的局部变量sum,最后主程序输出局部变量sum中的数值后结束程序运行。,7.2.7 注释语句虽然程序是由计算机执行的,但程序是由人设计的,另外,当程序出现问题或需要改进时,需要人在理解原设计思想和设计方法的基础上做出新的改进。这样,对人来说,程序是否容易理解就显得非常重要。我们把容易被人理解的程序称作可读性好的程序。设计可读性好程序的一个重要和基本的方法是在程序中添加足够的注释

40、信息,来说明设计该程序的思想和具体方法。因此,程序的基本元素包括注释语句。,不同的高级语言表示注释语句的方法不同。C语言注释语句的表示方法是:/*注释内容*/ 或/注释内容例如,在for循环语句后面加一个注释内容为“循环语句”的注释语句如下:for(n = 1; n = 100; n = n + 1) sum = sum + n; /循环语句 要说明的是,程序中的注释语句只是为提高可读性添加给人看的,计算机在执行程序时并不考虑注释语句。,7.3 高级语言的编译,7.3.1 编译的基本概念我们把高级语言形式的程序称为源程序,把机器语言形式的程序称为目标程序,编译程序是用来把源程序转换为等价的目标

41、程序的程序。编译程序相当于两种语言系统之间的“翻译”。用高级语言编写的源程序只有经过编译程序的“翻译”,变成了目标程序,计算机才可以识别和运行。,要在计算机上执行用高级语言编写的程序,一般需要经过编译和运行两个阶段。在编译阶段,计算机把高级语言形式的源程序编译为机器语言;在运行阶段,计算机运行目标程序,得到程序的处理结果。计算机执行高级语言程序的步骤如图7-8所示。,图7-8 计算机执行高级语言程序的步骤,图7-8中的计算机A和计算机B可以是同一台(或同一种类型)的计算机,也可以是不同类型的计算机。计算机在运行目标程序时,除需要目标程序外,还需要一些辅助程序配合,例如,数据格式转换程序、标准函

42、数程序等是这种辅助程序。因此,一个完整的编译软件包括编译程序和运行程序两大部分。一般用术语编译系统来表示完整的编译软件。,编译程序有两种“翻译”方式,一种是图7-8所示的完整的翻译后再执行的方式,完成这种翻译工作的程序称作编译程序。另一种是边翻译边执行的方式,完成这种翻译工作的程序称作解释程序。解释程序与编译程序的区别是,解释程序在翻译过程中不产生目标程序,而是边翻译边执行源程序本身。这种边翻译边执行的工作方式的最大缺点是效率太低,最大优点是简单易实现。一些功能较弱的简单高级语言(如BASIC 语言)采用这种解释方式,大部分高级语言都采用编译方式。现在也有一些高级语言采用编译方式和解释方式相结

43、合的工作方式。编译软件(或称编译系统)已成为目前计算机系统最重要的系统软件之一。,7.3.2 编译过程程序设计人员使用的高级语言是一个语言系统,机器语言是另一个语言系统,编译是把高级语言系统的源程序翻译成等价的机器语言系统的目标程序。因此,编译过程类同于人类之间两种语言系统(如英语到汉语)的翻译。一个翻译人员要把用英语写的文字翻译成意义相同的汉语的文字,他必须具备理解英文单词的能力,理解英文单词构成句子的规则,理解英文句子所对应的汉语含义。这三种能力是完成翻译工作的基础;要达到好的翻译(例如一个文学作品的翻译)效果,还要在上述初步翻译的基础上进行修辞加工,最后才能得到好的汉语翻译文字。,类似地

44、,编译程序要把用高级语言编写的源程序翻译成等价的机器语言形式的目标程序,编译程序就要能够识别出单词,掌握单词组成语句的规则,理解语句的含义,并要能够在此基础上,实现机器语言程序的优化,最后得到计算机可高效率执行的机器语言形式的目标程序。翻译过程和编译过程的对比见表7-4。,表7-4 翻译过程和编译过程的对比,从表7-4可见,编译过程至少包括五个子过程,即词法分析、语法分析、语义分析、代码优化和目标代码生成。这五个子过程不是互不相关的,而是紧密相关的,上一步工作是下一步工作的基础。因此,上一步程序要把工作的结果传送给下一步程序,编译过程还需要一个负责完成程序间信息传递工作的程序,这个程序在编译程

45、序中称作信息表管理程序。程序设计人员设计的源程序可能存在各种各样的错误,编译程序在进行上述编译工作时,每一个环节都可能发现源程序中存在的错误,所以,编译过程还要包括错误检查和处理程序。因此,完整的编译过程应该包括七个子过程,这七个子过程之间的逻辑关系如图7-9所示。,图7-9 编译过程的逻辑结构,词法分析、语法分析和语义分析是编译过程的基础。要使编译过程有效地进行,首先要解决的问题是如何确切地定义一种高级语言,其次要解决的问题是如何识别和分析这种高级语言。形式语言是解决上述两个问题的基础。1959年,语言学家乔姆斯基在对一些自然语言语法研究的基础上,提出了一种用来描述语言(包括高级程序设计语言

46、)的上下文无关文法。上下文无关文法是用一组数学符号和规则来描述语言的。利用上下文无关文法,对语言的分析过程变成了把具体的语句和该语言允许的文法规则进行匹配的过程。,上下文无关文法可以表示成语法图的形式。例如,C语言赋值语句的语法图如图7-10所示。C语言程序的任何一条语句只要能归纳成图7-10中最上面的符号形式,就认为该语句是一条正确的赋值语句。如语句sum=sum+n就是一条正确的赋值语句,而语句sum=sum#n就不是一条赋值语句,因为符号“#”使该语句和赋值语句语法图中的所有规则都不匹配。,图7-10 赋值语句的语法分析树,7.3.3 软件的运行经编译过程产生的目标程序还需要经过连接过程

47、和装载过程计算机才能执行。编译过程、连接过程和装载过程构成了软件的开发环境。下面简单地讨论这两个问题。1. 连接和装载从原理上说,源程序经过编译程序的编译就变成了计算机硬件可理解和执行的目标程序。但是一般较复杂的软件都是由许多模块组成的,这许多个模块分别经过编译后,产生的目标程序模块互不相连,所以需要考虑各模块间的联系问题。,另外,程序只有装入内存后才能运行,不同计算机的内存情况各异,就是同一台计算机在不同的时刻其内存使用情况也不同,所以需要考虑目标程序装入内存的方法问题。因此,源程序在经过了编译程序编译变成目标程序后,还需要解决运行目标程序所必须的模块间的联系问题和装入内存的方法问题。这两个

48、过程分别称作连接和装载。,连接就是把若干个分别编译完成的目标程序按照其调用关系连接起来(或者说组装起来),形成一个完整的模块。通常高级语言把经编译过程产生的程序模块命名为后缀为.obj的文件,把经连接过程产生的程序模块命名为后缀为.exe的文件。由于经连接过程产生的目标程序已是连接为一个整体的程序,所以,经连接过程产生的程序模块称作可执行程序。例如,若一个C语言的源程序文件名为exam.c,那么,经编译过程产生的文件名就是exam.obj,经连接过程产生的文件名就是exam.exe。程序exam.exe就是一个可执行程序。由于可执行程序都是以文件形式存放在磁盘等存储介质上,所以可执行程序也称作

49、可执行文件。,各个计算机在不同时刻的内存使用情况是不相同的,但是,整个可执行程序中各条机器指令的相对位置是固定不变的,只要程序装入内存单元的起始地址确定,整个可执行程序中各条机器指令的具体内存地址就可以确定。因此,可执行程序都是设计成可重定位方式的(即可按给出的内存起始地址确定各条机器指令在内存中的实际地址),装载过程根据当前计算机装入时所确定的起始地址把可执行程序装入内存。,2. 软件的开发环境高级语言程序需经过编译过程、连接过程和装载过程才能在计算机上运行,早先的编译系统中,编译子程序、连接子程序和装载子程序都是由用户根据需要输入不同的命令完成的。为了方便高级语言程序的设计过程,目前,基本上所有高级语言编译系统都把这些子程序整合在一个大的系统中,并把这样的包括了完整编译子程序、连接子程序、装载子程序等的系统称作开发环境。,另外,开发环境中还包括了其他一些子程序,例如开发环境中通常还包括源程序编辑子程序、文件保存子程序、源程序调试子程序等。开发环境中所有子程序都以菜单方式或图标方式提供给用户。用户在这样一个集成化的开发环境中,可根据自己当前设计工作的需要,选择相应的菜单或击活相应的图标来完成自己的设计工作。图7-11是Visual C+ 6.0开发环境的用户界面。其中,Edit菜单可用于编辑C语言或C+语言源程序,图7-11的文本框内显示的是当前正在编辑的源程序。,

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

当前位置:首页 > 企业管理 > 管理学资料

本站链接:文库   一言   我酷   合作


客服QQ:2549714901微博号:道客多多官方知乎号:道客多多

经营许可证编号: 粤ICP备2021046453号世界地图

道客多多©版权所有2020-2025营业执照举报