《第8章函数和编译预处理.ppt》由会员分享,可在线阅读,更多相关《第8章函数和编译预处理.ppt(105页珍藏版)》请在三一办公上搜索。
1、第8章 函数和编译预处理,C语言是一种结构化程序设计语言,而结化程序设计的总体思想是采用模块化结构,自上而下,逐步求精。即首先把一个复杂的大问题分解为若干相对独立的小问题,如果小问题仍较复杂,则可以把这些小问题又继续分解成若干子问题,这样不断地分解,使得小问题或子问题简单到能够直接用程序的三种基本结构表达为止。然后,对应每一个小问题或子问题编写出一个功能上相对独立的程序块来,这种像积木一样的程序块被称为模块。,前面各章中介绍的程序,都只有1个主函数main()。实际上,1个较大的应用程序,按结构化程序设计的要求,往往需要分成多个模块。C语言是通过函数实现模块化程序设计的,所以1个大的C语言程序
2、,是有多个函数组成的,每个函数分别对应各自的功能模块。在这些函数中,可以调用C编译系统提供的库函数,也可以调用自己编写的、或别人编写的自定义函数。函数的分类:从用户角度分为:由系统提供标准函数(库函数)和用户自定义函数从函数形式分为:无参函数和有参函数,8.1 函数的定义与调用,8.1.1函数的定义1任何函数(包括主函数main())都是由函数说明和函数体两部分组成。根据函数是否需要参数,可将函数分为无参函数和有参函数两种。(1)无参函数的一般形式 函数类型 函数名(void)说明语句部分;可执行语句部分;例如:void fun(void)printf(“C programn”);,(2)有参
3、函数的一般形式函数类型 函数名(数据类型 参数,数据类型 参数2)说明语句部分;可执行语句部分;有参函数比无参函数多了一个参数表。调用有参函数时,调用函数将赋予这些参数实际的值。为了与调用函数提供的实际参数区别开,将函数定义中的参数表称为形式参数表,简称形参表。(3)空函数既无参数、函数体又为空的函数。其一般形式为:函数类型 函数名(void),例题8.1 定义一个函数,用于求两个数中的大数。/*功能:定义一个求较大数的函数并在主函数中调用*/int max(int n1,int n2)/*定义一个函数max(),n1,n2为形参*/int z;z=n1n2?n1:n2;/*返回n1,n1中较
4、大者*/return(z)main()int max(int n1,int n2);/*函数说明与函数定义第一行一样写,但在最后加;*/int num1,num2;/*num1,num2是实际参数*/printf(input two numbers:n);scanf(%d%d,/*使程序暂停,看结果。按任一键继续*/,运行情况:6 9max=9,2说明:,(1)函数类型:指出return语句返回值的类型,它可以是C语言中任意合法的数据类型。如:int float char等,函数类型缺省时,系统默认为int 型。(2)函数名:是一个标识符。标识函数的名称。(3)函数名后括号内是形式参数,写出参
5、数的类型和名字。如:int max(int n1,int n2);不能写成:int max(int n1,n2);(4)在老版本C语言中,参数类型说明允许放在函数说明部分单独指定。例如:int max(n1,n2);int n1,n2;,(5)一个C程序由一个main主函数和多个子函数组成,执行从main函数开始,调用其他函数后,返回到main函数,在main函数中结束整个程序的运行。(6)函数定义不允许嵌套。在语言中,所有函数(包括主函数main())都是平行的。一个函数的定义,可以放在程序中的任意位置,主函数main()之前或之后。但在一个函数的函数体内,不能再定义另一个函数,即不能嵌套定
6、义。,8.1.2 函数的返回值与函数类型,语言的函数兼有其它语言中的函数和过程两种功能,从这个角度看,又可把函数分为有返回值函数和无返回值函数两种。1函数返回值与return语句有参函数的返回值,是通过函数中的return语句来获得的。当然有参函数如果不需要返回值,也可以没有return语句。(1)return语句的一般格式:return(返回值表达式);或return返回值表达式;(2)return语句的功能:返回调用函数,并将“返回值表达式”的值带给调用函数。(3)一个函数中可以有一个以上的return语句,执行到哪一个return语句,哪一个语句起作用。(4)被调用函数中可以无retur
7、n语句,当无return语句时并不是不返回一个值,而是一个不确定的值。为了明确表示不返回值,可以用“void”定义成“无(空)类型”。,例题8.2“void”定义的“无(空)类型”函数的举例void f1(int x,int y)/*定义f1函数,形参有整型的x、y,void表示空类型*/int w;w=x+y;printf(“w=%dn”,w);main()int a,b;a=3;b=4;f1(a,b);/*函数的调用,实参是整型的a、b*/,2函数类型在定义函数时,对函数类型的说明,应与return语句中返回值表达式的类型一致。如果不一致,则以函数类型为准。如果缺省函数类型,则系统一律按整
8、型处理。,例题8.3返回值类型和函数类型不同,以函数为准。/*例题源代码文件名:LT8_3.C*/int max(float n1,float n2)float z;z=n1n2?n1:n2;return(z);/*返回n1,n1中较大者*/main()float a,b;int c;scanf(“%f,%f”,运行情况:1.5 4.5max=4max函数中return(z);z返回值是float型,而函数定义返回值类型是int型,以函数为准,所以是int型,运行结果是max=4。,8.1.3 对被调用函数的说明和函数原型,1如果调用库函数,不用说明,但应该在本文件开头用#include命令将
9、调用有关库函数“包含”到本文件中。如:#include“stdio.h”在stdio.h文件中放了输入输出库函数所用到的一些宏定义信息。2如果调用自定义函数,在调用之前,应对被调用函数进行说明,其目的是:使编译系统知道被调用函数返回值的类型、函数参数的个数、类型、和顺序,便于调用时,对调用函数提供的实际参数的类型、个数、及顺序进行检查,看是否与被调用函数一致。,在ANSI C新标准中,采用函数原型方式,对被调用函数进行说明,其一般格式如下:函数类型 函数名(数据类型 参数名,数据类型 参数名2);例如:例8.1主函数中的int max(int n1,int n2);语句,是函数说明语句。说明函
10、数max的返回值类型为整型,有两个形式参数n1,n2都是整型。语言同时又规定,在以下2种情况下,可以省去对被调用函数的说明:(1)当被调用函数的函数定义出现在调用函数之前时。因为在调用之前,编译系统已经知道了被调用函数的函数类型、参数个数、类型和顺序。例8.1 max函数在主函数main()之前,主函数中的int max(int n1,int n2);说明语句可以不要。,(2)如果在所有函数定义之前,在函数外部(例如文件开始处)预先对各个函数进行了说明,则在调用函数中可缺省对被调用函数的说明。例如:char f1(int a);/*函数说明*/float f2(float b);/*函数说明*
11、/main()char f1(int a)/*函数定义*/float f2(float b);/*函数定义*/,8.1.4 函数的调用,在程序中,是通过对函数的调用来执行函数体的,其过程与其它语言的子程序调用相似。1函数调用语言中,函数调用的一般形式为:函数名(实际参数表)(1)实参的个数、类型和顺序,应该与被调用函数所要求的参数个数、类型和顺序一致,才能正确地进行数据传递。(2)如果实参有多个,对实际参数的求值顺序随系统而异。Turbo C按自右向左顺序求值。,例 8.4 函数实际参数的求值顺序。main()int f(int a,int b);/*函数说明*/int i=2,p;p=f(i
12、,+i);/*函数调用实参求值顺序是从右向左*/printf(“p=%d,p);int f(int a,int b)/*a=3 b=3*/int c;if(ab)c=1;else if(a=b)c=0;else c=-1;return(c);,运行结果:p=0,2函数调用方式,在语言中,可以用以下几种方式调用函数:(1)函数表达式。函数作为表达式的一项,出现在表达式中,以函数返回值参与表达式的运算。这种方式要求函数是有返回值的。如:c=2*max(a,b);(2)函数语句。C语言中的函数可以只进行某些操作而不返回函数值,这时的函数调用可作为一条独立的语句。如:max(a,b);(3)函数实参。
13、函数作为另一个函数调用的实际参数出现。这种情况是把该函数的返回值作为实参进行传送,因此要求该函数必须是有返回值的。,如:n=max(a,max(b,c);其中max(b,c)是一次调用,它的值作为max另一次调用的实参。(4)调用时实参与形参在类型必须匹配。如果类型不匹配,C编译程序将按赋值兼容的规则进行转换。(如:实参实型a=3.5,而形参x 为整型,则x得到的是3)。如果实参和形参的类型不赋值兼容,通常并不给出出错信息,且程序仍然继续执行,只是得不到正确的结果。,8.1.5 函数的形参与实参,函数的参数分为形参和实参两种,作用是实现数据传送。形参出现在函数定义中,只能在该函数体内使用。发生
14、函数调用时,调用函数把实参的值复制1份,传送给被调用函数的形参,从而实现调用函数向被调用函数的数据传送。关于形参与实参的说明:(1)实参可以是常量、变量、表达式、函数等。无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值,以便把这些值传送给形参。因此,应预先用赋值、输入等办法,使实参获得确定的值。(2)形参变量只有在被调用时,才分配内存单元;调用结束时,即刻释放所分配的内存单元。,因此,形参只有在该函数内有效。调用结束,返回调用函数后,则不能再使用该形参变量。(3)实参对形参的数据传送是单向的,即只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。(4)实参和形参占用不同
15、的内存单元,即使同名也互不影响。如以下程序:例题8.5 实参对形参的数据传递。/*功能:实参对形参的数据传递。*/*例题源代码文件名:LT8_5.C*/,main()void s(int n);/*说明函数*/int n=100;/*定义实参n,并初始化*/s(n);/*调用函数*/printf(n_s=%dn,n);/*输出调用后实参的值,便于进行比较*/void s(int n)int i;printf(n_x=%dn,n);/*输出改变前形参的值*/for(i=n-1;i=1;i-)n=n+i;/*改变形参的值*/printf(n_x=%dn,n);/*输出改变后形参的值*/,运行结果:
16、n_x=100 n_x=5050 n_s=100,8.2函数的嵌套调用和递归调用,8.2.1函数的嵌套调用 函数的嵌套调用是指,在执行被调用函数时,被调用函数又调用了其它函数。这与其它语言的子程序嵌套调用的情形是类似的,其关系可表示如图8-1。,例题8.6计算s=1k+2k+3k+N k/*例题源代码文件名:LT8_6.C*/*功能:函数的嵌套调用*/#define K 4#define N 5long f1(int n,int k)/*计算n的k次方*/long power=n;int i;for(i=1;ik;i+)power*=n;return power;,long f2(int n,
17、int k)/*计算1到n的k次方之累加和*/long sum=0;int i;for(i=1;i=n;i+)sum+=f1(i,k);return sum;main()printf(Sum of%d powers of integers from 1 to%d=,K,N);printf(%dn,f2(N,K);getch();,本程序编写了两个函数:一个用来计算n的函数f1(),另一个计算累加和的函数f2()。主函数先调用f2()计算累加和,再在f2()中调用f1()。f1()和f2()均为长整型,都在主函数之前定义,故不必再在主函数中对f1()和f2()加以说明。由于计算数值会很大,所以函
18、数和一些变量的类型都说明为长整型。,8.2.2 函数的递归调用函数的递归调用是指,一个函数在它的函数体内,直接或间接地调用它自身。,例8.7 用递归法计算n!。用递归法计算n!,可以用下面公式表示:1(n=1)n!=n*(n-1)!(n1)/*例题源代码文件名:LT8_7.C*/*功能:通过函数的递归调用计算阶乘*/long p(int n)long f;if(n1)f=p(n-1)*n;else f=1;return(f);,main()int n;long y;printf(input a inteager number:);scanf(%d,运行结果:input a inteager n
19、umber:55!=120,例题8.8 填空题:以下程序的输出结果是:_/*例题源代码文件名:LT8_8.C*/func(int x)int p;if(x=0|x=1)return(3);p=x-func(x-2);return p;main()printf(“%dn”,func(9);,分析:,所以程序的输出结果是:7,8.3 数组作为函数参数,数组用作函数参数有两种形式:一种是把数组元素(又称下标变量)作为实参使用;另一种是把数组名作为函数的形参和实参使用。8.3.1 数组元素作为函数参数数组元素就是下标变量,它与普通变量并无区别。数组元素只能用作函数实参,其用法与普通变量完全相同:在发生
20、函数调用时,把数组元素的值传送给形参,实现单向值传送。,例题8.9写一函数统计字符串中字母的个数。/*例题源代码文件名:LT8_9.C*/*功能:数组元素作为函数实参*/int isalp(char c)if(c=a,main()int i,num=0;char str255;printf(Input a string:);gets(str);for(i=0;stri!=0;i+)if(isalp(stri)num+;/*数组元素作为实际参数调用函数*/puts(str);/*输出字符串*/printf(num=%dn,num);/*输出字母个数*/,运行结果:Input a string:W
21、e study Turbo C!We study Turbo C!num=13,本例题子函数功能判断一个字符是否为字母,是返回值为1,否则为0。主函数调用时用语句for(i=0;stri!=0;i+)if(isalp(stri)num+;isalp(stri)返回值为1,表示是真,用num+统计字母的个数,isalp(stri)返回值为0,则什么也不做。说明:(1)用数组元素作实参时,只要数组类型和函数的形参类型一致即可,并不要求函数的形参也是下标变量。换句话说,对数组元素的处理是按普通变量对待的。(2)在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。
22、在函数调用时发生的值传送,是把实参变量的值赋予形参变量。,8.3.2 数组名作为函数的形参和实参,数组名作函数参数时,既可以作形参,也可以作实参。数组名作函数参数时,要求形参和相对应的实参都必须是类型相同的数组(或指向数组的指针变量),都必须有明确的数组说明。例题8.10已知某个学生5门课程的成绩,求平均成绩。/*例题源代码文件名:LT8_10.C*/float aver(float a)/*求平均值函数*/int i;float av,sum=a0;for(i=1;i5;i+)sum+=ai;av=sum/5;return av;,void main()float s5,av;int i;p
23、rintf(ninput 5 scores:n);for(i=0;i5;i+)scanf(%f,运行结果:input 5 scores:80 70 65 90 75average score is 76.00,说明:(1)用数组名作函数参数,应该在调用函数和被调用函数中分别定义数组,且数据类型必须一致,否则结果将出错。例如,在本案例中,形参数组为a,实参数组为s,它们的数据类型相同。(形参数组与实参数组可以同名)(2)C编译系统对形参数组大小不作检查,所以形参数组可以不指定大小。例如,本案例中的形参数组a。为了在被调用函数中处理数组元素的需要,可以另设一个参数,传递数组元素的个数。例如:将上例
24、中的aver()改进为如下:,例题8.11float aver(float a,int n)/*求平均值函数*/int i;float av,sum=a0;for(i=1;in;i+)sum+=ai;av=sum/n;return av;main()float s15=98.5,97,91.5,60,55;float s210=98,85,75,70,60,65,77,88,90,66;printf(“aver1=%6.2fn”,aver(s1,5);/*调用函数,实参为一数组名s1*/printf(“aver2=%6.2fn”,aver(s2,10);,运行结果:aver1=80.40ave
25、r2=77.4,可以看出,两次调用aver()函数时数组大小是不同的,在调用时用一个实参传递数组大小给形参n,以便aver()函数能处理数组大小不同的数组元素。如果指定形参数组的大小,则实参数组的大小必须大于等于形参数组,否则因形参数组的部分元素没有确定值而导致计算结果错误。,(3)数组名作为参数,只是将实参数组的首地址传给形参数组,从而使形参数组与实参数组共用同一段内存空间。假设s1数组的起始地址为1000,则a数组的起始地址也是1000显然形参数组和实参数组为同一数组。s0与a0同占一个单元。所以如果形参数组元素的值发生变化,也就是实参数组元素发生变化。这种数据传递方式称为“地址传递”。,
26、8.4 函数编程举例,1编程举例例8.12编程求sum=1+1/2+1/3+1/4+1/5+1/n/*例题源代码文件名:LT8_12.C*/#include“stdio.h”main()double fun(int n);/*函数的说明,注意末尾要加;*/int n;scanf(“%d”,int i;for(i=1;i=n;i+)sum+=1.0/i;return(sum);注意:本例计算1+1/2+1/31/n.由于“/”除号当除数和被除数均为整型时,商为整数,1/2=0、1/3=0所以要这样写sum+=1.0/i;被除数是实型1.0除数是整型作“/”除法sum才能得到实数各项和。,运行时输
27、入10输出结果为:sum=2.928968,2阅读程序训练,例8.13 以下程序运行结果是_10,20_.func(int a,int b)int temp;temp=a;a=b;b=temp;main()int x,y;x=10;y=20;func(x,y);printf(“%d,%dn”,x,y);(a)10,20(b)10,10(c)20,10(d)20,20,答案:(a),解析:这里是实参x,y向形参a,b值传递,在函数中只将形参a,b的值交换。不会改变实参的值,所以输出x,y的值仍为10,20。,例题8.14以下程序运行结果是_0 2 4 6_./*例题源代码文件名:LT8_14.C
28、*/void f1(int b)int j;for(j=0;j4;j+)bj=2*j;main()int a=5,6,7,8,i;f1(a);for(i=0;i4;i+)printf(“%dt”,ai);,解析:本题是数组名作为实际参数进行调用,只是将实参数组a的首地址传给形参数组b,从而使形参数组与实参数组共用同一段内存空间。即数组b就是数组a,所以形参数组b的变化就是a数组的变化,所以在主函数中的printf(“%dt”,ai);语句,输出0 2 4 6。这里的t是转义字符表示横向跳到下一个输出区,使用它为了隔开每一个数据。,8.5 内部变量与外部变量,语言中所有的变量都有自己的作用域。变
29、量说明的位置不同,其作用域也不同,据此将语言中的变量分为内部变量和外部变量。8.5.1内部变量在一个函数内部定义的变量是内部变量,它只在该函数范围内有效。也就是说,只有在本函数内部才能使用它们,在此函数之外就不能使用这些变量了。所以内部变量也称“局部变量”。,例如:int f1(int a)/*函数f1*/int b,c;/*a,b,c作用域:仅限于函数f1()中*/int f2(int x)/*函数f2*/int y,z;/*x,y,z作用域:仅限于函数f2()中*/main()int m,n;/*m,n作用域:仅限于函数main()中*/,关于内部变量的作用域还要说明以下几点:,1主函数m
30、ain()中定义的内部变量,也只能在主函数中使用,其它函数不能使用。同时,主函数中也不能使用其它函数中定义的内部变量。因为主函数也是一个函数,与其它函数是平行关系。这一点是与其它语言不同的,应予以注意。2形参变量也是内部变量,属于被调用函数;实参变量,则是调用函数的内部变量。3允许在不同的函数中使用相同的变量名,它们代表不同的对象,分配不同的单元,互不干扰,也不会发生混淆。4在复合语句中也可定义变量,其作用域只在复合语句范围内main()int a,b;.int c;/*c是在复合语句中定义的变量因此在复合语句范围内有效*/c=a+b;,8.5.2 外部变量,在函数外部定义的变量称为外部变量。
31、外部变量不属于任何一个函数,其作用域是:从外部变量的定义位置开始,到本文件结束为止。外部变量可被作用域内的所有函数直接引用,所以外部变量又称全局变量。如:int a=1,b=5;float f1(int x)int c,d;,char c1,c2;char f2(int x,int y)int i,j;main()int m,n;,对于外部变量还有以下几点说明:1.外部变量可加强函数模块之间的数据联系。由于同一文件中的所有函数都可以引用外部变量的值,因此,如果在一个函数中改变了外部变量的值,就能影响到其他函数,相当于各函数间有直接的传递通道。由于函数的调用只能带回一个返回值,因此有时可以利用外
32、部变量与函数联系的渠道,从函数得到一个以上的返回值。,例题8.15 输入长方体的长(l)、宽(w)、高(h),求长方体体积及正、侧、顶三个面的面积。/*例题源代码文件名:LT8_15.C*/。/*功能:利用外部变量计算长方体的体积及三个面的面积*/int s1,s2,s3;int vs(int a,int b,int c)int v;v=a*b*c;s1=a*b;s2=b*c;s3=a*c;return v;,main()int v,l,w,h;clrscr();printf(ninput length,width and height:);scanf(%d%d%d,运行结果:input le
33、ngth,width and height:3 4 5v=60 s1=12 s2=20 s3=15,本例程序的开头,定义了三个外部变量s1,s2,s3,存放长方体的正、侧、顶三个面的面积。其作用域是从开头到文件末尾,所以在主函数中仍有效,相当于把在子函数中的计算值带回主函数。长方体的体积v用return v;语句返回体积值。这样达到从子函数返回了一个以上的值。但是,以上方法虽然外部变量加强了函数模块之间的数据联系,但又使这些函数依赖这些外部变量,因而使得这些函数的独立性降低。从模块化程序设计的观点来看这是不利的,因为在程序设计中,模块的功能要单一,,与其他模块的互相影响要尽量少,而用外部变量是
34、不符合这个原则的。一般要求把c程序中的函数做成一个封闭体,除了可以通过“实参和形参“的渠道与外界发生联系外,没有其他渠道。这样的程序移值性好,可读性强。因此不是非用不可时,不要使用外部变量。2.在同一源文件中,允许外部变量和内部变量同名。在内部变量的作用域内,外部变量将被屏蔽而不起作用。例题8.16外部变量与内部变量同名/*例题源代码文件名:LT8_16.C*/,int a=3,b=5;/*a,b是外部变量*/max(int a,int b)/*a,b是函数的形参为内部变量*/int c;c=ab?a:b;return(c);main()int a=8;/*a为主函数的内部变量*/printf
35、(“%d”,max(a,b);,运行结果:8,本例在主函数中调用时,max(a,b)中,a是主函数的内部变量起作用,外部变量a将被屏蔽而不起作用。因此a=8 b=5进行调用,结果返回8。3.外部变量的作用域是从定义点到本文件结束。如果本文件定义点之前的函数需要引用这些外部变量或在其他文件中要引用时,需要用extern来声明外部变量,以扩展外部变量的作用域。外部变量说明的一般形式为:extern 数据类型 外部变量,外部变量2;,在一个文件内声明外部变量如果外部变量不在文件的开头定义,其有效的作用范围只限于定义处到文件末尾。如果本文件定义点之前的函数需要引用这些外部变量,则如下例方法进行引用。例
36、题8.17用extern声明外部变量,扩展程序文件中的作用域。/*例题源代码文件名:LT8_17.C*/,int max(int x,int y)int z;z=xy?x:y;return(z);main()extern a,b;/*在引用前外部变量声明*/printf(“%d”,max(a,b);/*引用外部变量*/int a=13,b=-8;/*定义外部变量在后*/,运行结果:13,(2)在多个文件的程序中声明外部变量在一个文件中定义外部变量,而另一个文件中用extern进行声明。例题8.18用extern将外部变量的作用域扩展到其他文件。文件LT8_18a.C的内容:int a;main
37、()int b,c,m=3;scanf(“%d%d”,文件LT8_18b.C的内容:extern a;/*声明本文件LT8_18b.C中的a是一个已经在其他文件中定义过的外部变量*/fun(int n)int y;y=a+n;retirn(y);用以上方法应十分慎重,因为在执行一个文件中的函数时,可能会改变该外部变量的值,它会影响到另一个文件的函数执行结果。,4在定义外部变量时加static,则外部变量只限于被本文件引用,而不能被其他文件引用。文件file1.c的内容:static int a;main()则a变量只能在文件file1.c中使用。(定义是不加static就可以用extern扩展
38、其作用域范围。如前面第3点所说),8.6 变量的存储类别,8.6.1动态存储方式和静态存储方式上一节从变量的作用域(即从空间)角度来分,分为:内部变量和外部变量。从另一角度,从变量值存在的时间(即生存期)角度来分,可分为:动态存储方式和静态存储方式。所谓静态存储方式是指在程序运行期间分配固定的存储空间的方式。而动态存储方式则是在程序运行期间根据需要进行动态的分配存储空间的方式。内存中供用户使用的存储空间情况如图8-51.程序区 2.静态存储区3.动态存储区,静态存储区存放外部变量。动态存储区存放以下数据:函数的形参。在调用函数时给形参分配存储空间。未加static声明的内部变量(即自动变量,后
39、面介绍)函数调用时的现场保护和返回地址。8.6.2自动变量函数中的内部变量,如果定义时不加static,就是自动变量。是动态分配存储空间的。函数的形参和在函数中定义的变量,都属于此类。在调用该函数时系统会给它们分配存储空间,在函数调用结束就自动释放这些存储空间。自动变量用关键字auto作存储类别的声明。,例如:int f(int a)auto int b,c=3;a是形参,b,c是自动变量,执行完f函数后,自动释放a,b,c所占的存储空间。关键字”auto”可以省略。不写auto 隐含表示是自动变量。auto int b,c;和int b,c;是等价的关于自动变量的说明:1自动变量属于动态存储
40、方式。在函数中定义的自动变量,只在该函数内有效;函数被调用时分配存储空间,调用结束就释放。,在复合语句中定义的自动变量,只在该复合语句中有效;退出复合语句后,也不能再使用,否则将引起错误。2定义而不初始化,则其值是不确定的。如果初始化,则赋初值操作是在调用时进行的,且每次调用都要重新赋一次初值。3由于自动变量的作用域和生存期,都局限于定义它的个体内(函数或复合语句),因此不同的个体中允许使用同名的变量而不会混淆。即使在函数内定义的自动变量,也可与该函数内部的复合语句中定义的自动变量同名。,8.6.3用static声明的内部变量,有时希望函数中的内部变量的值在函数调用结束后不消失而保留原值,在下
41、一次调用该函数时,该变量已有值,就是上一次函数调用结束的值。这时用“静态内部变量”。1定义静态内部变量方式:static 数据类型 内部变量表;存储特点:(1)静态内部变量属于静态存储。在程序执行过程中,即使所在函数调用结束也不释放。换句话说,在程序执行期间,静态内部变量始终存在,但其它函数是不能引用它们的,只能本函数引用。(2)定义但不初始化,则自动赋以(整型和实型)或0(字符型);,(3)对静态内部变量是在编译时赋初值的,即只赋初值一次,在程序运行时,它已有初值。以后每次调用它们所在的函数时,不再重新赋初值而只是保留上次函数调用结束时的值。例题8.19静态内部变量的使用。/*例题源代码文件
42、名:LT8_19.C*/f(int a)auto int b=0;/*动态内部变量(自动变量)*/static int c=3;/*静态内部变量*/b=b+1;,c=c+1;return(a+b+c);main()int a=2,x;for(x=0;x3;x+)printf(“%d”,f(a);,运行结果:7 8 9,例题8.19调用过程各变量值列表,8.6.4寄存器变量register,一般情况下,变量的值都是存储在内存中的。为提高执行效率,语言允许将局部变量的值存放到寄存器中,这种变量就称为寄存器变量。(寄存器存取速度快)定义格式如下:register 数据类型 变量表;(1)只有局部动态
43、变量和形参才能定义成寄存器变量,即全局变量不行。(2)对寄存器变量的实际处理,随系统而异。例如,微机上的MSC和TC 将寄存器变量实际当作自动变量处理。(3)允许使用的寄存器数目是有限的,不能定义任意多个寄存器变量。(4)局部静态变量不能定义为寄存器变量。,8.7 内部函数和外部函数,当一个源程序由多个源文件组成时,语言根据函数能否被其它源文件中的函数调用,将函数分为内部函数和外部函数。8.7.1 内部函数(又称静态函数)如果在一个源文件中定义的函数,只能被本文件中的函数调用,而不能被同一程序其它文件中的函数调用,这种函数称为内部函数。定义一个内部函数,只需在函数类型前再加一个“static”
44、关键字即可,如下所示:static 函数类型 函数名(函数参数表)关键字“static”,译成中文就是“静态的”,所以内部函数又称静态函数。但此处“static”的含义不是指存储方式,而是指对函数的作用域仅局限于本文件。,8.7.2 外部函数,如果在一个源文件中定义的函数,除可被本文件中的其他函数调用外,也可被其他文件中的函数所调用,这种函数称为外部函数。外部函数在整个源程序中都有效。外部函数的定义:在定义函数时,如果没有加关键字“static”,或冠以关键字“extern”,表示此函数是外部函数:extern 函数类型 函数名(函数参数表)调用外部函数时,需要对其进行说明:extern 函数
45、类型 函数名(参数类型表),函数名2(参数类型表2);,如:外部函数应用,(1)文件mainf.cmain()extern void input(),process(),output();/*对外部函数进行说明*/input();process();output();/*调用外部函数*/(2)文件subf1.cextern void input()/*定义外部函数*/(3)文件subf2.cextern void process()/*定义外部函数*/(4)文件subf3.cextern void output()/*定义外部函数*/,8.7.3 多个源程序文件的编译和连接,1、用Turbo
46、C集成环境(以上面4个文件组成的程序为例)(1)先后分别输入并编辑文件mainf.c,文件subf1.c,文件subf2.c,文件subf3.c存盘。(2)创建Project(项目)文件:用编辑源文件相同的方法,创建一个扩展名为.PRJ的项目文件,该文件中仅包括将被编译、连接的各源文件名,一行一个,其扩展名.C可以缺省;文件名的顺序,仅影响编译的顺序,与运行无关。项目文件的内容为:mainf.csubf1.csubf2.csubf3.c,(3)设置项目名称:打开菜单,选取Project菜单选Project name项按回车,屏幕出现一个对话框,询问项目文件名,输入项目文件名a.prj,其作用是
47、表示当前准备编译的是a.prj中包括的文件。(4)按功能键F9进行编译、连接,系统先后将4个文件翻译成目标文件,并把它们连接成一个可执行文件a.exe。(5)按ctrl+F9运行可执行文件a.exe。与单个源文件相同。编译产生的目标文件,以及连接产生的可执行文件,它们的主文件名,均与项目文件的主文件名相同。注意:当前项目文件调试完毕后,应选取ProjectClear project,将其项目名称从“Project name”中清除(清除后为空)。否则,编译、连接和运行的,始终是该项目文件!,8.8编译预处理,所谓编译预处理是指在对源程序进行编译之前,先对源程序中的编译预处理命令进行处理;然后再
48、将处理的结果,和源程序一起进行编译,以得到目标代码。编译预处理是C语言的一个重要功能,由编译预处理程序完成。为了与 C语言语句区别开,编译预处理命令以#号打头,单独占用一个书写行,行尾不使用分号作为结束符。,8.8.1 宏定义与宏展开,在语言中,“宏”分为无参数的宏(简称无参宏)和有参数的宏(简称有参宏)两种。1无参宏定义(1)无参宏定义的一般格式#define 标识符 语言符号字符串其中:“define”为宏定义命令;“标识符”为所定义的宏名,通常用大写字母表示,以便于与变量区别;“语言符号字符串”可以是常数、表达式等。,例题8.20输入圆的半径,求圆的周长、面积和球的体积。要求使用无参宏定
49、义圆周率。,#define PI 3.1415926/*PI是宏名,3.1415926用来替换宏名的常数*/main()float r,len,s,v;printf(Input a radius:);scanf(%f,编译预处理后,宏展开的结果如下:len=2*3.1415926*r;s=3.1415926*r*r;/*用3.1415926替换宏名PI*/v=3.1415926*r*r*r*3/4;,(2)说明,宏名一般用大写字母表示,以示与变量区别。但这并非是规定。宏定义不是语句,所以不能在行尾加分号。否则,宏展开时,会将分号作为字符串的1个字符,用于替换宏名。宏展开时,预处理程序仅以按宏定
50、义简单替换宏名,而不作任何检查。如果有错误,只能由编译程序在编译宏展开后的源程序时发现。如:#define PI 3.I4I59把数字1写成字母I,预处理时也照样代入,不管含义是否正确,不作任何检查。宏定义命令#define出现在函数的外部,宏名的有效范围是:从定义命令之后,到本文件结束。通常,宏定义命令放在文件开头处。如果需要,也可以#undef命令终止宏名的作用域。如:,#define PI 3.1415926main()#undef PI/*终止宏名PI的作用域,即PI只在main()函数中有效*/void fun()在进行宏定义时,可以引用已定义的宏名。#define PI 3.141