《C语言程序设计函数.pptx》由会员分享,可在线阅读,更多相关《C语言程序设计函数.pptx(110页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、6.1 概概 述述 函数是一段程序,它完成特定的任务,使用它时可用简单的方法为其提供必要的数据,然后自动执行这段程序,计算完毕后能保存计算结果回到程序原来的位置继续计算。第1页/共110页如果把编程比做制造一台机器,函数就好比其零部件可将这些“零部件”单独设计、调试、测试好,用时拿出来装配,再总体调试这些“零部件”可以是自己设计制造/别人设计制造/现在的标准产品而且,许多“零部件”我们可以只知道需向它提供什么,它能产生什么,并不需要了解它是如何工作、如何设计制造的所谓“黑盒子”。黑箱(函数)结果(输出参数)输入参数第2页/共110页函数用于把较大的计算任务分解成若干个较小的任务使程序人员可以在
2、其他函数的基础上构造程序,而不需要从头做起一个设计得当的函数可以把具体操作细节对程序中不需要知道它们的那些部分隐藏掉,从而使整个程序结构清楚,减轻了因修改程序所带来的麻烦。C语言在设计函数时考虑了效率与易于使用这两个方面。一个C程序一般都由许多较小的函数组成,而不是只由几个比较大的函数组成。一个程序可以驻留在一个文件中,也可以存放在多个文件中。各个文件可以单独编译并与库中已经编译过的函数装配在一起。第3页/共110页编写一个测试软件显示软件封面检查密码产生题目接受回答评判计分显示结果如果要继续练习告别词第4页/共110页main()charans=y;clrscr();cover();/*调用
3、软件封面显示函数*/password();/*调用密码检查函数*/while(ans=y|ans=Y)question();/*调用产生题目函数*/answers();/*调用接受回答函数*/marks();/*调用评分函数*/results();/*调用结果显示函数*/printf(“是否继续练习?(Y/N)n”);ans=getchar();printf(“谢谢使用,再见!”);第5页/共110页/*定义所用函数*/cover()/*软件封面显示函数*/password()/*密码检查函数*/question()/*产生题目函数*/answers()/*接受回答函数*/marks()/*评
4、分函数*/results()/*结果显示函数*/第6页/共110页在C程序设计中,通常:将一个大程序分成几个子程序模块(自定义函数)将常用功能做成标准模块(标准函数)放在函数库中供其他程序调用将程序中反复使用的程序段写成函数的形式,某些只用一次的程序段往往也写成函数形式 引入函数的优点:减少重复编写程序的工作量使程序便于调试和阅读第7页/共110页例6.1main()/*主函数*/printstar();/*调用printstar函数画*/print_message();/*调用print_message函数写字*/printstar();/*调用printstar函数画*/printstar
5、();/*printstar函数*/printf(“n*”);print_message();/*print_message函数*/printf(“nHello!”);运行结果:*Hello!*第8页/共110页说明:程序的执行从main函数开始,由主函数调用其它函数,其它函数也可以相互调用,程序流程最后回到main函数,在main函数中结束整个程序的运行mainmain函数是系统定义的,只能由系统调用函数是系统定义的,只能由系统调用所有函数在定义时都是平行的,相互独立(一个函数并不从属于另一个函数),即函数不能嵌套定义,但可以相互调用一个程序可以由一个或多个源程序文件组成第9页/共110页一
6、个(主调)函数可以多次调用多个(被调)函数。同一个函数也可以被一个或多个(主调)函数调用任意多次。下图说明一种调用关系:mainabcfdefghid第10页/共110页函数分类使用情况库函数(标准函数):由系统提供,用户可以直接使用用户自定义函数:解决用户的专门需要函数形式无参函数:主调函数与被调函数之间没有数据传递有参函数:主调函数与被调函数之间有数据传递任务情况带返回值不带返回值第11页/共110页第12页/共110页函数的定义函数的定义 函数名:标识符,用于标识函数,并用其来调用函数。函数名字有值,它代表函数的入口地址。形式参数表:说明参数的个数和类型,简称形参表。一般来说,计算函数需
7、要多少原始数据,函数的形参表中就有多少个形参,每个形参存放一个数据。变量说明:说明函数中用到的除形参以外的其它变量。函数体:为了完成特定的功能而设计的一个或多个语句。第13页/共110页第14页/共110页第15页/共110页intmax(x,y)intx,y;intz;z=xy?x:y;return(z)数据类型函数名(形参表)形参类型说明;说明部分语句intmax(intx,inty)intz;z=xy?x:y;return(z)/*现代风格是:函数名(带类型形参表)*/第16页/共110页第17页/共110页第18页/共110页第19页/共110页第20页/共110页第21页/共110页
8、为什么?第22页/共110页6.3 函数的调用函数的调用 函数定义制造函数函数使用声明(准备使用)调用(使用函数)通过函数调用,两个函数中的数据发生联系。第23页/共110页6.3.1 形式参数和实际参数形式参数和实际参数形式参数:定义函数时函数名后括号内的变量名称实际参数:主调函数中调用函数时,函数名后括号内的参数(变量,表达式等)第24页/共110页第25页/共110页说明说明形参在函数调用前不占用存储单元,在调用时被分配,调用后释放所占用的存储单元;形参只能是变量,而实参可以是常量、变量或表达式,但要求它们有确定的值。调用时将实参的值赋给形参。在函数中必须指定形参的类型;实参和形参个数相
9、等,按顺序对应,一一传递,实参和形参的类型应相同或赋值兼容;第26页/共110页语言规定,实参对形参的数据传递是“值传递”,即单向传递,只由实参传给形参,而不能由形参传回来给实参。在内存中,实参单元与形参单元是不同的单元。void fun(int i,int jint i,int j)int x=7;printf(i=%d,j=%d,x=%dn,i,j,x);main()int i=2,x=5,j=7;fun(j,6j,6);printf(i=%d,j=%d,x=%dn,i,j,x);i=7,j=6,x=7i=2,j=7,x=5第27页/共110页#includevoidsum(intx,in
10、ty,intz)z=x+y;main()inta=1,b=2,c=0;sum(a,b,c);printf(c=%dn,c);c=?0第28页/共110页6.3.2 函数的返回值函数的返回值函数的返回值是通过函数中的return语句获得的。return语句将被调用函数中的一个确定值带回主调函数中去。return语句一般形式:return(表达式);或:return表达式;或:return;第29页/共110页若return后面带表达式,首先计算表达式的值,表达式的值就是所求的函数值。表达式的类型与函数首部说明的类型一致。intint max(int x,int y)int z;z=(xy)?x:
11、y;return(z z);return(xy?x:yxy?x:y);一个函数可以含有多个return语句,但当执行到其中一个return语句就返回主调函数。一个函数可以没有return语句,此时当函数执行到最后一个界限符“”时返回主调函数。说明第30页/共110页函数值的类型:应当在定义函数时指定函数的类型。intintmax(intx,inty)doubledoublemin(intx,inty)charcharletter(charc1,charc2)max(intx,inty)intmax(intx,inty)语言规定,凡不加类型声明的函数,自动按语言规定,凡不加类型声明的函数,自动按
12、 intint型型处理。处理。第31页/共110页在定义函数时,对函数值声明的类型一般应该和return语句中的表达式类型一致;若不一致,则以函数类型为准(自动转换)。对于有返回值的函数,若return语句后面没有表达式,或没有return语句,此时带回一个不确定的返回值。为了明确表示“不带回值”,可以用void定义“无类型”(或称“空类型”)。void stars(int n)第32页/共110页printstar()printf(*);main()inta;a=printstar();printf(%d,a);例 函数带回不确定值输出:10voidprintstar()printf(*);
13、main()inta;a=printstar();printf(%d,a);编译错误!第33页/共110页第34页/共110页6.3.3 函数调用的一般形式函数调用的一般形式 函数名(实参1,实参2,实参n)执行过程:计算各个表达式,得到值v1,vn,把v1,vn赋给对应的形参。然后转去执行函数体,函数体执行结束后,即遇到return语句或执行完函数的最后一个语句,返回到函数调用处。第35页/共110页说明说明 实参与形参多个实参间用逗号隔开实参与形参间个数相等,类型应一致实参与形参按顺序对应,一一传递数据实参表求值的顺序与系统有关第36页/共110页调用方式:函数语句:由函数调用加上分号构成
14、,在主调函数中可作为一个独立的语句stars(20);print_message();函数表达式:函数调用作为一个运算对象出现在表达式中,此时要求函数带回一个确定的值以参加表达式的运算c=max(a,b);函数参数:函数调用作为另一个函数的实参,其值作为一个实际参数传给被调函数的形参进行处理;此时也要求函数带回一个确定值m=max(a,max(b,c);第37页/共110页调用位置首先被调用函数必须是已存在的函数,如用户自定义函数或库函数。如果使用库函数,需要在文件的开头用#include命令将需要的库函数包含到文件中。#include调用同一源文件中的非标准函数时,必须在主调函数中对所调函数
15、进行声明:函数声明语句(函数原型)第38页/共110页6.4 函数原型函数原型一般形式:类型标识符 被调函数名(形参类型 形参名);功能:通知编译程序函数值是什么类型,有多少参数及它们各自的类型,为编译程序进行类型检查提供依据例如:double power(int x,int n);或 double power(x,n);或 double power();第39页/共110页例6.5main()floatadd();floata,b,c;scanf(“%f,%f”,&a,&b);c=printf(“sumis%f”,add(a,b);floatadd(x,y)floatx,y;floatz;z
16、=x+y;return(z);/*对被调用函数的说明*/作为表达式被调用/*定义add函数*/运行结果:3.6,5.5sumis9.100000第40页/共110页函数定义和函数声明的区别函数定义和函数声明的区别函数的定义是确定函数的功能,包括函数名,函数值类型,形参及类型和函数体全部内容。函数的声明只是对要被调用的函数的返回值的类型进行说明,它只包括函数名、函数类型或形参类型,不包括函数体。第41页/共110页三种情况下可以省略声明:函数值是整型(int)或字符型(char)时系统自动按整型说明;所调函数的定义出现在主调函数之前时;文件一开头,在所有函数之前,对所用函数作了声明第42页/共1
17、10页无函数说明整型函数main()inta,b,c;scanf(“%d,%d”,&a,&b);c=max(a,b);printf(“Maxis%d”,c);max(x,y)intx,y;intz;z=xy?x:y;return(z);函数值是整型(int)或字符型(char)时系统自动按整型说明第43页/共110页被调函数在主调函数之前主调函数在被调函数之后被调用函数出现在主调用函数之前。floatadd(floatx,floaty)floatz;z=x+y;return(z);main()floata,b;scanf(%f,%f,&a,&b);printf(sumis%f,add(a,b)
18、;第44页/共110页在所有函数之前说明函数类型此处不必说明定义letter函数文件一开头,对所用函数作了声明charletter();main().charletter(c1,c2)charc1,c2;.第45页/共110页第46页/共110页第47页/共110页附:调用外部函数(其他源文件中定义的函数)时 函数说明语句 extern函数名();例文件file1.c中main()intx=80,y=90,c;externmax();/*函数说明*/c=max(x,y)+20;/*调用max函数*/printf(“Maxis%dn”,c);文件files2.c中(与file1.c同目录)ext
19、ernmax(inta,intb)/*extern可省略*/floatc;c=ab?a:b;returnc;第48页/共110页6.5 函数的嵌套调用函数的嵌套调用 C语言允许嵌套调用函数,即在调用一个函数的过程中,又调用另一个函数主函数()声明a函数调用a函数a函数()声明b函数调用b函数b函数()第49页/共110页6.6 函数的递归调用函数的递归调用 C语言允许递归调用函数,即在调用一个函数的过程中,又直接或间接的调用该函数本身直接递归调用调用函数的过程中又调用该函数本身intf(intx)inty;y=f(x);returny;第50页/共110页间接递归调用调用f1函数的过程中调用f
20、2函数,而f2中又需要调用f1。显然,以上均为无终止递归调用。为此,一般要用if语句来控制使递归过程到某一条件满足时结束。intf1(intx)inty;y=f2(x);returny;intf2(intx)inty;y=f1(x);returny;第51页/共110页递归法:类似于数学证明中的反推法,从后一结果与前一结果的关系中寻找其规律性。归纳法可以分为:递推法 从初值出发,归纳出新值与旧值间直到最后值为止存在的关系要求通过分析得到:初值+递推公式编程:通过循环控制结构实现(循环的终值是最后值)递归法 从结果出发,归纳出后一结果与前一结果直到初值为止存在的关系要求通过分析得到:初值+递归函
21、数第52页/共110页编程:设计一个函数(递归函数),这个函数不断使用下一级值调用自身,直到结果已知处选择控制结构其一般形式是:在主函数中用终值n调用递归函数,而在递归函数中:递归函数名f(参数x)if(n=初值)结果=;else结果=含f(x-1)的表达式;返回结果(return);第53页/共110页第54页/共110页实际上,递归程序分两个阶段执行(1)回推(调用):欲求n!先求(n-1)!(n-2)!1!若1!已知,回推结束。(2)递推(回代):知道1!2!可求出3!n!main()intn;floats;floatfac();printf(Inputn=);scanf(%d,&n);
22、s=fac(n);printf(%d!=%.0f,n,s);floatfac(intx)intf;if(x=0|x=1)f=1;elsef=fac(x-1)*x;returnf;运行:Inputn=55!=120第55页/共110页第56页/共110页图示第57页/共110页程序:main()clrscr();printf(%d,age(5);age(intn)intc;if(n=1)c=10;elsec=age(n-1)+2;returnc;运行:18第58页/共110页11112113311464115101051分析:第x行有x个值对第x行第y列,其值(不计左侧空格时)1(y=1或y=x
23、)c(x,y)=c(x-1,y-1)+c(x-1,y)例例6.8在屏幕上显示杨辉三角形在屏幕上显示杨辉三角形第59页/共110页程序如下:main()inti,j,n;clrscr();printf(Inputn=);scanf(%d,&n);for(i=1;i=n;i+)for(j=0;j=n-i;j+)printf();for(j=1;j=i;j+)printf(%3d,c(i,j);printf(n);intc(intx,inty)intz;if(y=1|y=x)return1;elsez=c(x-1,y-1)+c(x-1,y);returnz;第60页/共110页#includerev
24、()charc;c=getchar();if(c=$)printf(%c,c);elserev();printf(%c,c);main()rev();例例6.9运行下列程序,当输入字符序列运行下列程序,当输入字符序列AB$CDE并回车时,程序的输出结果是什么并回车时,程序的输出结果是什么?运行:$BA第61页/共110页例例6.10反向输出一个整数(非数值问反向输出一个整数(非数值问题)题)非数值问题的分析无法象数值问题那样能得出一个初值和递归函数式,但思路是相同的。分析方法:1)简化问题:设要输出的正整数只有一位,则“反向输出”问题可简化为输出一位整数。2)对大于10的正整数,逻辑上可分为两
25、部分:个位上的数字和个位以前的全部数字。将个位以前的全部数字看成一个整体,则为了反向输出这个大于10的正整数,可按以下步骤:a、输出个位上的数字;b、将个位除外的其他数字作为一个新的整数,重复a步骤的操作。第62页/共110页其中b问题只是对原问题在规模上进行了缩小递归。所以,可将反向输出一个正整数的算法归纳为:if(n为一位整数)输出n;else输出n的个位数字;对剩余数字组成的新整数重复“反向输出”操作;第63页/共110页程序如下:#includevoidmain()voidprintn(intx);intn;printf(Inputn=);scanf(%d,&n);if(n=0&x%c
26、n,getone,putone);voidhanoi(intn,charone,chartwo,charthree)if(n=1)move(one,three);elsehanoi(n-1,one,three,two);move(one,three);hanoi(n-1,two,one,three);main()intm;printf(Inputthenumberofdisks:);scanf(%d,&m);printf(Thestepstomoving%3ddisks:n,m);hanoi(m,A,B,C);Inputthenumberofdisks:3Thestepstomoving3di
27、sks:A-CA-BC-BA-CB-AB-CA-C第67页/共110页递归的条件:1、须有完成函数任务的语句;longf(intn)longy;if(n=1)y=1;elsey=n*f(n-1);return(y);2、一个确定是否能避免递归调用的测试;3、一个递归调用语句;该语句的参数应该逐渐逼近结束条件,以至最后断绝递归。4、先测试,后递归调用。在递归函数定义中,必须先测试,后递归调用。也就是说,递归调用是有条件的,满足了条件后,才可以递归。longf(intn)longy;y=n*f(n-1);if(n=1)y=1;return(y);第68页/共110页递归的特点:1、递归调用不是重新
28、复制该函数,每次调用它时,新的局部变量和形参会在内存中重新分配内存单新的局部变量和形参会在内存中重新分配内存单元元,并以新的变量重新开始执行;每次递归返回时,当前调用层的局部变量和形参被释放,并返回上次调用自身的地方继续执行;2、递归调用一般并不节省内存空间,因为每次调用都要产生一组新的局部变量,从而不破坏上层的局部变量;3、递归调用一般并不能加快程序的执行速度,因为每次调用都要保护上层局部量(现场),而返回时又要恢复上层局部量,占用执行时间;4、递归函数中,必须有结束递归的条件;5、递归调用的优点是能实现一些迭代算法难以解决的问题。第69页/共110页6.7 数组作为函数参数数组作为函数参数
29、数组元素可以作函数的实参数组名可以作函数的实参和形参多维数组可以作函数参数6.7.1数组元素作函数的实参由于表达式可以做实参,数组元素可以作为表达式的组成部分,因此,数组元素可以做函数的实参,并且可以单向传递给形参。第70页/共110页例 两个数组大小比较432105a562312107688432105b212343986654n=0m=0k=0in=0m=0k=1in=0m=1k=1in=1m=1k=1in=1m=1k=2in=2m=1k=2in=3m=1k=2a和b为有10个元素的整型数组比较两数组对应元素变量n,m,k记录aibi,ai=bi,aik,认为数组ab 若nk,认为数组ab
30、若n=k,认为数组a=b#includemain()printf(Enterarraya:n);for(i=0;i10;i+)scanf(%d,&ai);printf(Enterarrayb:n);for(i=0;i10;i+)scanf(%d,&bi);for(i=0;iy)flag=1;elseif(xy)flag=-1;elseflag=0;return(flag);第71页/共110页6.7.2数组名作函数参数地址传递在主调函数与被调函数分别定义数组,且类型应一致形参数组大小(多维数组第一维)可不指定形参数组名是地址变量第72页/共110页例 求学生的平均成绩#include floa
31、t average(int stu10,int n);void main()int score10,i;float av;printf(Input 10 scores:n);for(i=0;i10;i+)scanf(%d,&scorei);av=average(score,10);printf(Average is:%.2f,av);float average(int stu10,int n)int i;float av,total=0;for(i=0;in;i+)total+=stui;av=total/n;return av;实参用数组名形参用数组定义,intstu.2109score56
32、2312.88stu第73页/共110页说明:1、用数组名作函数参数,应在主调函数和被调函数中用数组名作函数参数,应在主调函数和被调函数中分别定义数组,而不能只在一方定义,被调函数是在分别定义数组,而不能只在一方定义,被调函数是在形参表中进行定义;形参表中进行定义;floatscore10;aver=average(score);floataverage(floatarray10)2、形参数组和实参数组的大小可一致可不一致;语形参数组和实参数组的大小可一致可不一致;语言编译时对形参数组大小不作检查,只将实参数组的言编译时对形参数组大小不作检查,只将实参数组的首地址传给形参数组;首地址传给形参数
33、组;3、形参数组可以不指定大小,仅给出数组类型、数形参数组可以不指定大小,仅给出数组类型、数组名和一对方括号,另设一个参数传递数组的实组名和一对方括号,另设一个参数传递数组的实际长度;际长度;floataverage(floatarray,intn)aver=average(score,10);第74页/共110页4 4、实参数组与形参数组类型应一致,否则出错;、实参数组与形参数组类型应一致,否则出错;5 5、数组名作函数实参时,不是把数组的值传递给形、数组名作函数实参时,不是把数组的值传递给形参,而是把实参数组的起始地址传递给形参数组;参,而是把实参数组的起始地址传递给形参数组;这样,形参数
34、组和实参数组就共占同一段内存单这样,形参数组和实参数组就共占同一段内存单元,即表示同一个数组。因此,形参数组元素的元,即表示同一个数组。因此,形参数组元素的变化等同于实参数组元素的变化;变化等同于实参数组元素的变化;6 6、从传递方式来看,应理解为、从传递方式来看,应理解为“传值传值”方式,即传方式,即传递的是数组名所代表的值。递的是数组名所代表的值。第75页/共110页例 数组元素与 数组名 作函数参数比较12a调用前a0a112a调用a0a112xy21xy交换12a返回#includevoidswap2(intx,inty)intz;z=x;x=y;y=z;main()inta2=1,2
35、;swap2(a0,a1);printf(a0=%dna1=%dn,a0,a1);值传递第76页/共110页12a调用前12ax调用21ax交换21a返回#includevoidswap2(intx)intz;z=x0;x0=x1;x1=z;main()inta2=1,2;swap2(a);printf(a0=%dna1=%dn,a0,a1);地址传递例 数组元素与 数组名 作函数参数比较第77页/共110页例例6.12用选择法对数组中用选择法对数组中1010个整数按个整数按由小到大排序由小到大排序所谓选择法就是:先将10个整数中最小的数与a0对换,再将a1到a9中最小的数与a1对换;.每比较
36、一轮,找出未经排序的数中最小的一个,共应比较 9 轮。程序设计函数sort(array,n)对数组元素按由小到大排序主程序:输入array数组,调用sort函数比较,输出排序后的array数组第78页/共110页排序函数:无返回值,数组名为形参形参数组说明可以不指定大小交换arrayj和arrayi;voidsort(intarray,intn)intv,j,t;for(i=0;in-1;i+)for(j=i+1;jn;j+)if(arrayjarrayi)t=arrayi;arrayi=arrayj;arrayj=t;第79页/共110页main()inta10,i;printf(“ente
37、rarray:n”);for(i=1;i10;i+)scanf(“%d”,&ai);sort(a,10);printf(“thesortedarray:n”);for(i=1;i10;i+)printf(“%4d”,ai);printf(“n”);实参数组说明数组a赋值调用排序函数由于地址传递,实参数组a改变第80页/共110页6.7.3多维数组作函数参数多维数组作函数参数1.形参数组定义时可以指定每一维的大小,也可省略第一维的大小说明(但不能省略第二维及其它高维的大小说明)例如:intarray310;或intarray10;但不能写成intarray3;和intarray;2.实参数组可以
38、大于形参数组例如:实参数组定义为:intarray510;形参数组定义为:intarray310;形参数组只取实参数组的一部分,其余部分不起作用第81页/共110页例写函数打印n行10列二维数组,其中n是任意整数voidprint_array(inta10,intn)inti,j;for(i=0;in;i+)for(j=0;j10;j+)printf(“%3d”,aij);printf(“n”);例写函数交换任意一个n*10的二维数组的i,j两行voidexchange(inta10,inti,intj)intk,t;for(k=0;k10;k+)t=aik;aik=ajk;ajk=t;第82
39、页/共110页例8.12n行10列int型数组的每一行都有一个最大值,写一函数,求这n个最大值的最小值.max_element(inta,intm)intmax,i;max=a0;for(i=1;imax)max=ai;returnmax;max_min(inta,intn)inti,min,max;min=max_element(a0,10);for(i=1;imax)min=max;returnmin;第83页/共110页例 求二维数组中最大元素值1 3 5 72 4 6 815 17 34 12ijmax=11 3 5 72 4 6 815 17 34 12ijmax=31 3 5 72
40、 4 6 815 17 34 12ijmax=5j1 3 5 72 4 6 815 17 34 12imax=7j1 3 5 72 4 6 815 17 34 12imax=7j1 3 5 72 4 6 815 17 34 12imax=34int max_value(int array34)int i,j,k,max;max=array00;for(i=0;i3;i+)for(j=0;jmax)max=arrayij;return(max);main()int a34=1,3,5,7,2,4,6,8,15,17,34,12;printf(max value is%dn,max_value(a
41、);多维形参数组第一维维数可省略,第二维必须相同intarray4第84页/共110页例 求二维数组中各行元素之和get_sum_row(intx3,intresult,introw,intcol)inti,j;for(i=0;irow;i+)resulti=0;for(j=0;jcol;j+)resulti+=xij;main()inta23=3,6,9,1,4,7;intsum_row2,row=2,col=3,i;get_sum_row(a,sum_row,row,col);for(i=0;irow;i+)printf(Thesumofrow%d=%dn,i+1,sum_rowi);31
42、4679asum_rowxresult1812第85页/共110页6.8 变量的作用域与存储类变量的作用域与存储类别别作用域定义:指定义:指变量能够起作用的程序范围。如果一个变量变量能够起作用的程序范围。如果一个变量在某源程序文件或某函数范围内有效,则称该文件或在某源程序文件或某函数范围内有效,则称该文件或函数为该变量的作用域。函数为该变量的作用域。局部变量局部变量全局变量全局变量从作用域角度从作用域角度生存期定义:指变量在内存中存在的时间范围。定义:指变量在内存中存在的时间范围。静态变量静态变量动态变量动态变量从生存期角度从生存期角度变量在整个程序的运行时间都存在变量只在某个函数的执行过程中
43、才存在第86页/共110页6.8.1 局部变量与全局变量局部变量与全局变量局部变量函数内部或复合语句内定义的变量称为局部变量例:int f1(int a)int f1(int a)int v1,x;int v1,x;int f2(int f2(voidvoid)int v2,x,y;int v2,x,y;main()main()int m,y;int m,y;a、v1、x的有效范围v2、x、y的有效范围m、y的有效范围第87页/共110页说明:、主函数、主函数mainmain中定义的变量也只在主函数中有效,其中定义的变量也只在主函数中有效,其它函数不能引用;它函数不能引用;、不同函数中可以使用
44、相同名字的变量,它们代表不、不同函数中可以使用相同名字的变量,它们代表不同的对象,占用不同的内存单元,互相独立;同的对象,占用不同的内存单元,互相独立;、形式参数也是局部变量;、可以在复合语句中定义变量,其作用域只是本复合语句。main()inta,b;intc;c=a+b;c在此范围内有效a、b在此范围内有效第88页/共110页void main()int i,a=0;for(i=1;i=2;i+)int a=1;a+;printf(i=%d,a=%dn,i,a);printf(i=%d,a=%dn,i,a);i=1,a=2i=2,a=2i=3,a=0全局变量定义:在函数外部定义的变量,又称
45、为外部变量。全局变量可以为本文件中其它函数所共用其有效范围为从定义变量的位置开始到本源文件结束第89页/共110页例:#include int p=1,q=5;float f1(int a)int b,c;char c1,c2;char f2(int x,int y)int i,j;.main()int m,n;全局变量p、q的作用域全局变量c1、c2的作用域第90页/共110页说明:、全局变量增加了函数间的数据联系;、尽量少使用全局变量(除非在必要时);各模块间的相互联系、相互影响太多,降低了模块的独立性;会降低程序的清晰性,因为各个函数都有可能改变全局变量的值,需要时刻记住变量的当前值,编
46、程时候容易出错。#include int k;void show()for(k=1;k=10;k+)putchar(*);putchar(n);main()main()for(k=1;k=4;k+)for(k=1;k=4;k+)show();show();第91页/共110页、若全局变量与局部变量同名,则在局部变量的作用范围内,全局变量不起作用;#include int a=3,b=5;int max(int a,int b)int c;c=ab?a:b;return(c);void main()int a=8;printf(%d,max(a,b);形参a、b的作用域全局变量a、b不起作用局部
47、变量a的作用域全局变量b的作用域全局变量a不起作用讨论:如果主函数中没有int a=8,结果?第92页/共110页、若全局变量在文件开头定义,则在整个程序中都可以使用;若不在开头定义,其作用域只限于说明处到文件结束。如果想在定义之前的函数中引用该全局变量,则在函数中用关键字“extern”作“外部变量声明”,在函数内部,从声明之处起,可以使用它们。#include int max(int x,int y)int z;z=xy?x:y;return(z);void main()extern int a,b;printf(%d,max(a,b);int a=13,b=-8;第93页/共110页6.
48、8.2 变量的存储类型变量的存储类型 结构化设计要求研究不同模块(函数、源文件)间变量的关系。变量两大属性:数据类型存储类别1、数据类型(复习)通过变量声明(定义)来规定其数据类型:格式数据类型关键字变量名如chara;intb,c;floatx,y;第94页/共110页预留存储空间(如char型为1个字节,int型为2个字节)确定存储方式(如char型存放ASCII值,int型存放补码值)char型01100001(用一个字节存放该字符的ASCII值)int型0110000111100110(用两个字节存放该数值的补码)float型0110101010011000011(四个字节)一个变量的
49、数据类型定义后,就规定了该变量只能存储相应类型的数据第95页/共110页2、存储空间的划分程序区静态存储区动态存储区程序区:用于存放程序编译后形成的可执行代码(执行时装入)静态存储区:用于存放程序中的静态数据,如全局变量等动态存储区:用于存放程序中的动态数据,如函数形参、局部变量、函数调用时的现场保护和返回地址等静态数据说明时在静态存储区中分配存储单元并在程序执行过程中始终占用该单元,直到程序结束才释放;动态数据在函数开始执行时分配动态存储空间,函数结束时释放这些空间。第96页/共110页3、存储类别规定了变量在计算机内部的存放位置决定变量的“寿命”(何时“生”,何时“灭”)一个完整的变量说明
50、格式如下:存储类别数据类型变量名如staticintx,y;的存储类别有四种:auto、static、register和extern。register型(寄存器型)变量值存放在运算器的寄存器中存取速度快,一般只允许23个,且限于char型和int型,通常用于循环变量第97页/共110页auto型(自动变量型)变量值存放在主存储器的动态存储区(堆栈方式)优点同一内存区可被不同变量反复使用static型(静态变量型)变量值存放在主存储器的静态存储区程序执行开始至结束,始终占用该存储空间extern型(外部变量型)同上,其值可供其他源文件使用以上两种均属于“动态存储”性质,即调用函数时才为这些变量分