【回顾】C程序设计第七章笔记
C程序设计(谭浩强第五版)第七章笔记

用函数实现模块化程序设计
1.为什么要用函数?
如果执行这个:
int main()
{
{
⋮ 功能1内容
}
{
⋮ 功能2内容
}
{
⋮ 功能1内容
}
{
⋮ 功能2内容
}
}
代码太冗长。难阅读,难修改。
改成下面的形式:
功能1函数()
{
⋮ 功能1内容
}
功能2函数()
{
⋮ 功能2内容
}
int main()
{
调用功能1
调用功能2
调用功能1
调用功能2
}
结构精炼,易于阅读,易于修改。结构清晰基于它的结构化或者说是模块化设计。
我们总结一下:
函数的特点:
①.使用函数可使程序清晰、精炼、简单、灵活。
②.函数就是功能。每一个函数用来实现一个特定的功能。函数名应反映其代表的功能。
③.在设计较大程序时,往往把它分为若干个程序模块,每一个模块包括一个或多个函数,每个函数实现一个特定的功能。
④.一个C程序可由一个主函数和若干个其他函数构成。由主函数调用其他函数,其他函数也可以互相调用。
再来回答为什么要用函数:
(1) 一个C程序由一个或多个程序模块组成,每一个程序模块作为一个源程序文件。较大的程序,可分别放在若干个源文件中。这样便于分别编写和编译,提高调试效率。一个源程序文件可以为多个C程序共用。
(2) 一个源程序文件由一个或多个函数以及其他有关内容(如指令、数据声明与定义等)组成。一个源程序文件是一个编译单位,在程序编译时是以源程序文件为单位进行编译的,而不是以函数为单位进行编译的。
(3) C程序的执行是从main函数开始的,如果在main函数中调用其他函数,在调用后流程返回到main函数,在main函数中结束整个程序的运行。
(4) 所有函数都是平行的,即在定义函数时是分别进行的,是互相独立的。一个函数并不从属于另一个函数,即函数不能嵌套定义。函数间可以互相调用,但不能调用main函数。main函数是被操作系统调用的。
(5) 从用户使用的角度看,函数有两种。
① 库函数,它是由系统提供的,用户不必自己定义,可直接使用它们。应该说明,不同的C语言编译系统提供的库函数的数量和功能会有一些不同,当然许多基本的函数是共同的。
② 用户自己定义的函数。它是用以解决用户专门需要的函数。
(6) 从函数的形式看,函数分两类。
① 无参函数。在调用无参函数时,主调函数不向被调用函数传递数据。
② 有参函数。主调函数在调用被调用函数时,通过参数向被调用函数传递数据。
2.定义函数
C语言要求,在程序中用到的所有函数,必须“先定义,后使用”。
定义函数应包括以下几个内容:
(1) 指定函数的名字,以便以后按名调用。
(2) 指定函数的类型,即函数返回值的类型。
(3) 指定函数的参数的名字和类型,以便在调用函数时向它们传递数据。对无参函数不需要这项。
(4) 指定函数应当完成什么操作,也就是函数是做什么的,即函数的功能。这是最重要的,是在函数体中解决的。
定义函数的方法(三种:无参函数,有参函数,空函数。这里没有分析返回值类型)
(1).定义无参函数
类型名 函数名()
{
函数体
}
或者
类型名 函数名(void)
{
函数体
}
函数名后面括号内的void表示“空”,即函数没有参数。
函数体包括声明部分和语句部分。
在定义函数时要用“类型标识符”(即类型名)指定函数值的类型,即指定函数带回来的值的类型。
(2).定义有参函数
类型名 函数名(形式参数表列)
{
函数体
}
(3).定义空函数
类型名 函数名()
{ }
作用:函数体为空,什么也不做。
3.调用函数
3.1函数调用的形式
语法:函数名(实参表列)
(1). 函数调用语句
把函数调用单独作为一个语句。如printf_star();
这时不要求函数带回值,只要求函数完成一定的操作。
(2). 函数表达式
函数调用出现在另一个表达式中,如c=max(a,b);
这时要求函数带回一个确定的值以参加表达式的运算。
(3). 函数参数
函数调用作为另一个函数调用时的实参。如m=max(a,max(b,c));,又如:printf (″%d″, max (a,b));
3.2.形式参数和实际参数
①.在调用有参函数时,主调函数和被调用函数之间有数据传递关系。
②.在定义函数时函数名后面括号中的变量名称为“形式参数”(简称“形参”)或“虚拟参数”。
③.在主调函数中调用一个函数时,函数名后面括号中的参数称为“实际参数”(简称“实参”)。 实际参数可以是常量、变量或表达式,但要求它们有确定的值。
④.实参与形参的类型应相同或赋值兼容。赋值兼容是指实参与形参类型不同时能按不同类型数值的赋值规则进行转换。
3.3.函数调用的过程
(1) 在定义函数中指定的形参,在未出现函数调用时,它们并不占内存中的存储单元。在发生函数调用时,函数的形参才被临时分配内存单元。
(2) 将实参的值传递给对应形参。
(3) 在执行函数期间,由于形参已经有值,就可以利用形参进行有关的运算。
(4) 通过return语句将函数值带回到主调函数。应当注意返回值的类型与函数类型一致。如果函数不需要返回值,则不需要return语句。这时函数的类型应定义为void类型。
(5) 调用结束,形参单元被释放。注意: 实参单元仍保留并维持原值,没有改变。如果在执行一个被调用函数时,形参的值发生改变,不会改变主调函数的实参的值。因为实参与形参是两个不同的存储单元。
注意:
①.实参向形参的数据传递是“值传递”,单向传递,只能由实参传给形参,而不能由形参传给实参。
②.实参和形参在内存中占有不同的存储单元,实参无法得到形参的值。
3.4.函数的返回值
通常,希望通过函数调用使主调函数能得到一个确定的值,这就是函数值(函数的返回值)。
(1) 函数的返回值是通过函数中的return语句获得的。一个函数中可以有一个以上的return语句,执行到哪一个return语句,哪一个return语句就起作用。return语句后面的括号可以不要,如“return z;”与“return(z);”等价。return后面的值可以是一个表达式。
(2) 函数值的类型。函数值的类型在定义函数时指定。
(3) 在定义函数时指定的函数类型一般应该和return语句中的表达式类型一致。
如果函数值的类型和return语句中表达式的值不一致,则以函数类型为准。对数值型数据,可以自动进行类型转换。即函数类型决定返回值的类型。
(4) 对于不带回值的函数,应当用定义函数为“void类型”(或称“空类型”)。这样,系统就保证不使函数带回任何值,即禁止在调用函数中使用被调用函数的返回值。此时在函数体中也没有return语句。
3.5对被调用函数的声明和函数原型
在一个函数中调用另一个函数(即被调用函数)需要具备如下条件:
(1) 被调用的函数必须是已经定义的函数(是库函数或用户自己定义的函数)。
(2) 如果使用库函数,应该在本文件开头用#include指令将调用有关库函数时所需用到的信息“包含”到本文件中来。
(3) 如果使用用户自己定义的函数,而该函数的位置在调用它的函数(即主调函数)的后面(在同一个文件中),应该在主调函数中对被调用的函数作声明(declaration)。声明的作用是把函数名、函数参数的个数和参数类型等信息通知编译系统,以便在遇到函数调用时,编译系统能正确识别函数并检查调用是否合法。
3.6函数的嵌套调用
注意:
C语言的函数定义是互相平行、独立的,也就是说,在定义函数时,一个函数内不能再定义另一个函数,即不能嵌套定义,
但可以嵌套调用函数,即在调用一个函数的过程中,又调用另一个函数。
3.7函数的递归调用
在调用一个函数的过程中又出现直接或间接地调用该函数本身,称为函数的递归调用。
注意:
程序中不应出现无终止的递归调用,而只应出现有限次数的、有终止的递归调用,
这可以用if语句来控制,只有在某一条件成立时才继续执行递归调用;否则就不再继续。
4.数组作为函数参数
4.1.数组元素作为函数实参
数组元素可以用作函数实参,但是不能用作形参。
因为形参是在函数被调用时临时分配存储单元的,不可能为一个数组元素单独分配存储单元(数组是一个整体,在内存中占连续的一段存储单元)。
在用数组元素作函数实参时,把实参的值传给形参,是“值传递”方式。
数据传递的方向是从实参传到形参,单向传递。
4.2.一维数组名作函数参数
注意:
用数组元素作实参时,向形参变量传递的是数组元素的值,而用数组名作函数实参时,向形参(数组名或指针变量) 传递的是数组首元素的地址。
一维数组名作函数参数说明:
(1) 用数组名作函数参数,应该在主调函数和被调用函数分别定义数组。
(2) 实参数组与形参数组类型必须一致。
(3) 在定义average函数时,声明形参数组的大小为10,但在实际上,指定其大小是不起任何作用的,因为C语言编译系统并不检查形参数组大小,只是将实参数组的首元素的地址传给形参数组名。
(4) 形参数组可以不指定大小,在定义数组时在数组名后面跟一个空的方括号。
4.3.多维数组名作函数参数
可以用多维数组名作为函数的实参和形参,在被调用函数中对形参数组定义时可以指定每一维的大小,也可以省略第一维的大小说明。
5.局部变量和全局变量
每一个变量都有一个作用域问题,即它们在什么范围内有效。
5.1局部变量
定义变量可能有3种情况:
(1) 在函数的开头定义;
(2) 在函数内的复合语句内定义;
(3) 在函数的外部定义。
在一个函数内部定义的变量只在本函数范围内有效,也就是说只有在本函数内才能引用它们,在此函数以外是不能使用这些变量的。
在复合语句内定义的变量只在本复合语句范围内有效,只有在本复合语句内才能引用它们。
在该复合语句以外是不能使用这些变量的,以上这些称为“局部变量”。
(1) 主函数中定义的变量也只在主函数中有效。主函数也不能使用其他函数中定义的变量。
(2) 不同函数中可以使用同名的变量,它们代表不同的对象,互不干扰。
(3) 形式参数也是局部变量。只在定义它的函数中有效。其他函数中不能直接引用形参。
(4) 在一个函数内部,可以在复合语句中定义变量,这些变量只在本复合语句中有效,这种复合语句也称为“分程序”或“程序块”。
5.2.全局变量
程序的编译单位是源程序文件,一个源文件可以包含一个或若干个函数。
在函数内定义的变量是局部变量,而在函数之外定义的变量称为外部变量,外部变量是全局变量(也称全程变量)。
全局变量可以为本文件中其他函数所共用。它的有效范围为从定义变量的位置开始到本源文件结束。
作用:
设置全局变量的作用是增加了函数间数据联系的渠道。
由于同一文件中的所有函数都能引用全局变量的值,因此如果在一个函数中改变了全局变量的值,就能影响到其他函数中全局变量的值。
相当于各个函数间有直接的传递通道。由于函数的调用只能带回一个函数返回值,因此有时可以利用全局变量来增加函数间的联系渠道,通过函数调用能得到一个以上的值。
*为了便于区别全局变量和局部变量,在C程序设计人员中有一个习惯(但非规定),将全局变量名的第1个字母用大写表示。
但是,建议不在必要时不要使用全局变量,原因如下:
① 全局变量在程序的全部执行过程中都占用存储单元,而不是仅在需要时才开辟单元。
② 它使函数的通用性降低了,因为如果在函数中引用了全局变量,那么执行情况会受到有关的外部变量的影响,如果将一个函数移到另一个文件中,还要考虑把有关的外部变量及其值一起移过去。但是若该外部变量与其他文件的变量同名时,就会出现问题。这就降低了程序的可靠性和通用性。在程序设计中,在划分模块时要求模块的“内聚性”强、与其他模块的“耦合性”弱。即模块的功能要单一(不要把许多互不相干的功能放到一个模块中),与其他模块的相互影响要尽量少,而用全局变量是不符合这个原则的。一般要求把C程序中的函数做成一个相对的封闭体,除了可以通过“实参—形参”的渠道与外界发生联系外,没有其他渠道。这样的程序移植性好,可读性强。
③ 使用全局变量过多,会降低程序的清晰性,人们往往难以清楚地判断出每个瞬时各个外部变量的值。由于在各个函数执行时都可能改变外部变量的值,程序容易出错。因此,要限制使用全局变量。
6.变量的存储方式和生存期
动态存储方式与静态存储方式
从变量值存在的时间(即生存期)来观察,有的变量在程序运行的整个过程都是存在的,而有的变量则是在调用其所在的函数时才临时分配存储单元,而在函数调用结束后该存储单元就马上释放了,变量不存在了。
也就是说,变量的存储有两种不同的方式: 静态存储方式和动态存储方式。
静态存储方式是指在程序运行期间由系统分配固定的存储空间的方式。
动态存储方式则是在程序运行期间根据需要进行动态的分配存储空间的方式。
7.存储类别
在C语言中,每一个变量和函数都有两个属性: 数据类型和数据的存储类别。
存储类别指的是数据在内存中存储的方式(如静态存储和动态存储)。
在定义和声明变量和函数时,一般应同时指定其数据类型和存储类别,也可以采用默认方式指定(即如果用户不指定,系统会隐含地指定为某一种存储类别)。
C的存储类别包括4种: 自动的(auto)、静态的(statis)、寄存器的(register)、外部的(extern)。根据变量的存储类别,可以知道变量的作用域和生存期。
7.1.自动变量(auto变量)
(1).函数中的局部变量,如果不专门声明为static(静态)存储类别,都是动态地分配存储空间的,数据存储在动态存储区中。
(2).函数中的形参和在函数中定义的局部变量(包括在复合语句中定义的局部变量),都属于此类。
(3).在调用该函数时,系统会给这些变量分配存储空间,在函数调用结束时就自动释放这些存储空间。
(4).因此这类局部变量称为自动变量。自动变量用关键字auto作存储类别的声明。
实际上,关键字auto可以省略,不写auto则隐含指定为“自动存储类别”,它属于动态存储方式。程序中大多数变量属于自动变量。
7.2.静态局部变量(static局部变量)
(1) 静态局部变量属于静态存储类别,在静态存储区内分配存储单元。在程序整个运行期间都不释放。而自动变量(即动态局部变量)属于动态存储类别,分配在动态存储区空间而不在静态存储区空间,函数调用结束后即释放。
(2) 对静态局部变量是在编译时赋初值的,即只赋初值一次,在程序运行时它已有初值。以后每次调用函数时不再重新赋初值而只是保留上次函数调用结束时的值。而对自动变量赋初值,不是在编译时进行的,而是在函数调用时进行的,每调用一次函数重新给一次初值,相当于执行一次赋值语句。
(3) 如果在定义局部变量时不赋初值的话,则对静态局部变量来说,编译时自动赋初值0(对数值型变量)或空字符′\0′(对字符变量)。而对自动变量来说,它的值是一个不确定的值。这是由于每次函数调用结束后存储单元已释放,下次调用时又重新另分配存储单元,而所分配的单元中的内容是不可知的。
(4) 虽然静态局部变量在函数调用结束后仍然存在,但其他函数是不能引用它的。因为它是局部变量,只能被本函数引用,而不能被其他函数引用。
7.3.寄存器变量(register变量)
一般情况下,变量(包括静态存储方式和动态存储方式)的值是存放在内存中的。当程序中用到哪一个变量的值时,由控制器发出指令将内存中该变量的值送到运算器中。 经过运算器进行运算,如果需要存数,再从运算器将数据送到内存存放。
如果有一些变量使用频繁(例如,在一个函数中执行10 000次循环,每次循环中都要引用某局部变量),则为存取变量的值要花费不少时间。为提高执行效率,允许将局部变量的值放在CPU中的寄存器中,需要用时直接从寄存器取出参加运算,不必再到内存中去存取。由于对寄存器的存取速度远高于对内存的存取速度,因此这样做可以提高执行效率。
这种变量叫做寄存器变量,用关键字register作声明。由于现在的计算机的速度愈来愈快,性能愈来愈高, 优化的编译系统能够识别使用频繁的变量,从而自动地将这些变量放在寄存器中,而不需要程序设计者指定。因此,现在实际上用register声明变量的必要性不大。
全局变量都是存放在静态存储区中的。因此它们的生存期是固定的,存在于程序的整个运行过程。
一般来说,外部变量是在函数的外部定义的全局变量,它的作用域是从变量的定义处开始,到本程序文件的末尾。
在此作用域内,全局变量可以为程序中各个函数所引用。但有时程序设计人员希望能扩展外部变量的作用域。
这就引出了我们的extern关键字。
7.4.外部变量(extern变量)
(1).在一个文件内扩展外部变量的作用域
如果外部变量不在文件的开头定义,其有效的作用范围只限于定义处到文件结束。
在定义点之前的函数不能引用该外部变量。如果由于某种考虑,在定义点之前的函数需要引用该外部变量,
则应该在引用之前用关键字extern对该变量作“外部变量声明”,表示把该外部变量的作用域扩展到此位置。
有了此声明,就可以从“声明”处起,合法地使用该外部变量。
(2).将外部变量的作用域扩展到其他文件
如果一个程序包含两个文件,在两个文件中都要用到同一个外部变量Num,
不能分别在两个文件中各自定义一个外部变量Num,否则在进行程序的连接时会出现“重复定义”的错误。
正确的做法是: 在任一个文件中定义外部变量Num,而在另一文件中用extern对Num作“外部变量声明”,即“extern Num; ”。
在编译和连接时,系统会由此知道Num有“外部链接”,可以从别处找到已定义的外部变量Num,并将在另一文件中定义的外部变量Num的作用域扩展到本文件,在本文件中可以合法地引用外部变量Num。
extern既可以用来扩展外部变量在本文件中的作用域,又可以使外部变量的作用域从一个文件扩展到程序中的其他文件,系统在编译过程中遇到extern时,
先在本文件中找外部变量的定义,如果找到,就在本文件中扩展作用域;
如果找不到,就在连接时从其他文件中找外部变量的定义。如果从其他文件中找到了,就将作用域扩展到本文件;
如果再找不到,就按出错处理。
8.内部函数和外部函数
函数本质上是全局的,因为定义一个函数的目的就是要被另外的函数调用。
如果不加声明的话,一个文件中的函数既可以被本文件中其他函数调用,也可以被其他文件中的函数调用。
但是,也可以指定某些函数不能被其他文件调用。
根据函数能否被其他源文件调用,将函数区分为内部函数和外部函数。
8.1.内部函数
语法:
static 类型名 函数名(形参表);
例如:
static int fun(int a,int b)//表示fun是一个内部函数,不能被其他文件调用
内部函数又称静态函数,因为它是用static声明的。使用内部函数,可以使函数的作用域只局限于所在文件。这样,在不同的文件中即使有同名的内部函数,也互不干扰,不必担心所用函数是否会与其他文件模块中的函数同名。
通常把只能由本文件使用的函数和外部变量放在文件的开头,前面都冠以static使之局部化,其他文件不能引用。这就提高了程序的可靠性。
8.2.外部函数
例如:
extern 类型名 函数名(形参表);
extern int fun(int a,int b)//表示fun可以被其他文件调用
使用extern声明就能够在本文件中调用在其他文件中定义的函数,或者说把该函数的作用域扩展到本文件。
extern声明的形式就是在函数原型基础上加关键字extern。
由于函数在本质上是外部的,在程序中经常要调用其他文件中的外部函数,为方便编程,C语言允许在声明函数时省写extern。
用函数原型能够把函数的作用域扩展到定义该函数的文件之外(不必使用extern)。
只要在使用该函数的每一个文件中包含该函数的函数原型即可。函数原型通知编译系统: 该函数在本文件中稍后定义,或在另一文件中定义。
利用函数原型扩展函数作用域最常见的例子是#include指令的应用。在#include指令所指定的“头文件”中包含调用库函数时所需的信息。
第七章完!