收藏 分销(赏)

C语言-函数公开课获奖课件.pptx

上传人:精**** 文档编号:7916348 上传时间:2025-01-26 格式:PPTX 页数:119 大小:479.04KB
下载 相关 举报
C语言-函数公开课获奖课件.pptx_第1页
第1页 / 共119页
C语言-函数公开课获奖课件.pptx_第2页
第2页 / 共119页
点击查看更多>>
资源描述
单击此处编辑母版标题样式,单击此处编辑母版文本样式,第二级,第三级,第四级,第五级,单击此处编辑母版标题样式,单击此处编辑母版文本样式,第二级,第三级,第四级,第五级,第八章,第1页,8.1,概述,一种较大程序可分为若干个程序模块,每一种模块用来实现一种特定功能。在高级语言中用子程序实现模块功能。子程序由函数来完毕。一种程序可由一种主函数和若干个其他函数构成。,由主函数调用其他函数,其他函数也可以互相调用。同一种函数可以被一种或多种函数调用任意一再。,函数间调用关系,第2页,#include,void main(),void printstar();,/*对printstar函数申明*/,void print_message();,/*对print_message函数申明*/,printstar();,*调用printstar函数*,print_message();,/*调用print_message函数*/,printstar();,*调用printstar函数*/,例8.1先举一种函数调用简朴例子,第3页,void printstar(),*定义printstar函数*,printf(*n);,void print_message(),*定义print_message函数*,printf(How do you do!n);,运行状况如下:,*,How do you do!,*,第4页,阐明:,(1)一种程序由一种或多种程序模块构成,每一种程序模块作为一种源程序文献。对较大程序,一般不但愿把所有内容全放在一种文献中,而是将他们分别放在若干个源文献中,再由若干源程序文献构成一种C程序。这样便于分别编写、分别编译,提高调试效率。一种源程序文献可认为多种C程序公用。,第5页,(2)一种源程序文献由一种或多种函数以及其他有关内容(如命令行、数据定义等)构成。一种源程序文献是一种编译单位,在程序编译时是以源程序文献为单位进行编译,而不是以函数为单位进行编译。,(3)程序执行是从函数开始,如是在函数中调用其他函数,在调用后流程返回到函数,在函数中结束整个程序运行。,第6页,(4)所有函数都是平行,即在定义函数时是分别进行,是互相独立。一种函数并不从属于另一函数,即函数不能嵌套定义。函数间可以互相调用,但不能调用函数。函数是系统调用。,第7页,(5)从顾客使用角度看,函数有两种:,原则函数,即库函数。这是由系统提供,顾客不必自己定义这些函数,可以直接使用它们。应当阐明,不一样样C系统提供库函数数量和功能会有某些不一样样,当然许多基本函数是共同。,顾客自己定义函数。用以处理顾客专门需要。,第8页,(6)从函数形式看,函数分两类:,无参函数。如例8.1中printstar和print_message就是无参函数。在调用无参函数时,主调函数不向被调用函数传递数据。无参函数一般用来执行指定一组操作。例如,例8程序中printstar函数。,有参函数。在调用函数时,主调函数在调用被调用函数时,通过参数向被调用函数传递数据,一般状况下,执行被调用函数时会得到一种函数值,供主调函数使用。,第9页,8.函数定义一般形式,8.2.1.无参函数定义一般形式,定义无参函数一般形式为:,类型标识符函数名(),申明部分,语句部分,在定义函数时要用“类型标识符”指定函数值类型,即函数带回来值类型。例8.中printstar和print_message函数为void类型,表达不需要带回函数值。,第10页,8.2.2.有参函数定义一般形式,定义有参函数一般形式为:,类型标识符函数名(形式参数表列),申明部分,语句部分,例如:,(int,int),;/*函数体中申明部分*,?;,();,第11页,8,.,2,.,3,空函数,定义空函数一般形式为:,类型标识符函数名(),例如:,(),调用此函数时,什么工作也不做,没有任何实际作用。在主调函数中写上“();”表明“这里要调用一种函数”,而目前这个函数没有起作用,等后来扩充函数功能时补充上。,第12页,8,.函数参数和函数值,8,.形式参数和实际参数,在前面提到有参函数中,在定义函数时函数名背面括弧中变量名称为“形式参数”(简称“形参”),在主调函数中调用一种函数时,函数名背面括弧中参数(可以是一种表达式)称为“实际参数”(简称“实参”)。return背面括弧中值()作为函数带回值(称函数返回值)。,第13页,在不一样样函数之间传递数据,可以使使用方法:,参数:通过形式参数和实际参数,返回值:用return语句返回计算成果,全局变量:外部变量,大多数状况下,主调函数和被调用函数之间有数据传递关系。,第14页,#include,void(),int max(int,int);,/*对函数申明*/,int,;,scanf(,);,(,);,printf(,);,例8.调用函数时数据传递,第15页,int max(int,int),*定义有参函数max*,int;,?;,return();,运行状况如下:,,,第16页,通过函数调用,使两个函数中数据发生联络,第17页,有关形参与实参阐明:,(1)在定义函数中指定形参,在未出现函数调用时,它们并不占内存中寄存单元。只有在发生函数调用时,函数中形参才被分派内存单元。在调用结束后,形参所占内存单元也被释放。,(2)实参可以是常量、变量或表达式,如:,(,);,但规定它们有确定值。在调用时将实参值赋给形参。,第18页,(3)在被定义函数中,必须指定形参类型(见例8.2程序中,“(,);”,)。,(4)实参与形参类型应相似或赋值兼容。例8中实参和形参都是整型。假如实参为整型而形参x为实型,或者相反,则按第3章简介不一样样类型数值赋值规则进行转换。,例如实参值a为3.5,而形参x为整型,则将实数3.5转换成整数3,然后送到形参b。字符型与整型可以互相通用。,第19页,(5)在语言中,实参向对形参数据传递是“值传递”,单向传递,只由实参传给形参,而不能由形参传回来给实参。在内存中,实参单元与形参单元是不一样样单元。,第20页,在调用函数时,给形参分派寄存单元,并将实参对应值传递给形参,调用结束后,形参单元被释放,实参单元仍保留并维持原值。因此,在执行一种被调用函数时,形参值假如发生变化,并不会变化主调函数实参值。例如,若在执行函数过程中和值变为和,而和仍为和。,第21页,8.3.2 函数返回值,一般,但愿通过函数调用使主调函数能得到一种确定值,这就是函数返回值。例如,例8.中,(,)值是,(,)值是5。赋值语句将这个函数值赋给变量。,有关函数返回值某些阐明:,(1)函数返回值是通过函数中return语句获得。,第22页,假如需要从被调用函数带回一种函数值供主调函数使用,被调用函数中必须包括return语句。假如不需要从被调用函数带回函数值可以不要return语句。,一种函数中可以有一种以上return语句,执行到哪一种return语句,哪一种语句起作用。return语句背面括弧也可以不要,,如:“return;”等价于“return();”,return语句将被调用函数中一种确定值带回主调函数中去。见图8.2中从return语句返回箭头。,第23页,return背面值可以是一种表达式。,例如,例8中函数可以改写成:,(int,int),(,?);,(2)函数返回值应当属于某一种确定类型,在定义函数时指定函数返回值类型。,第24页,例如:下面是3个函数首行:,int max(float,float)/*函数值为整型*/,char letter(char c1,char c2)/*函数值为字符型*/,double min(int,int)/*函数值为双精度型*/,在语言中,凡不加类型阐明函数,自动按整型处理。例8.中函数首行函数类型int可以省写,用Turbo C 2.0编译程序时能通过,但用Turbo C 3.0编译程序时不能通过,由于C+规定所有函数都必须指定函数类型。因此,提议在定义时对所有函数都指定函数类型。,第25页,(3)在定义函数时指定函数类型一般应当和return语句中表达式类型一致。,假如函数值类型和return语句中表达式值不一致,则以函数类型为准。对数值型数据,可以自动进行类型转换。即函数类型决定返回值类型。,(4)对于不带回值函数,应当用“void”定义函数为“无类型”(或称“空类型”)。这样,系统就保证不使函数带回任何值,即严禁在调用函数中使用被调用函数返回值。此时在函数体中不得出现return语句。,第26页,例 8.返回值类型与函数类型不一样样,#include,void main(),int(float,float);,float,;,int;,scanf(,);,(,);,printf(,);,int max(float,float),float;,/*z为实型变量*/,?;,return();,运行状况如下:,,,Max is ,第27页,8.函数调用,8.函数调用一般形式,函数调用一般形式为:函数名(实参表列),假如实参表列包括多种实参,则各参数间用逗号隔开。实参与形参个数应相等,类型应匹配。实参与形参按次序对应,一一传递数据。,假如是调用无参函数,则“实参表列”可以没有,但括弧不能省略。,第28页,假如实参表列包括多种实参,对实参求值次序并不是确定,有系统按自左至右次序求实参值,有系统则按自右至左次序。许多版本是按自右而左次序求值,例如Tubro+。,第29页,例 8 实参求值次序,#include,void main(),int f(int a,int b);,/*函数申明*/,int i=2,p;,p=f(i,+i);,/*函数调用*/,printf(%dn,p);,第30页,int f(int a,int b),/*函数定义*/,int c;,if(ab)c=1;,else if(a=b)c=0;,else c=-1;,return(c);,第31页,假如按自左至右次序求实参值,则函数调用相称于(,),假如按自左至右次序求实参值,则函数调用相称于(3,),对于函数调用,int i=2,p;,p=f(i,+i);,第32页,8.函数调用方式,按函数在程序中出现旳位置来分,可以有如下三种函数调用方式:,函数语句,把函数调用作为一种语句。如例8.1中printstar(),这时不规定函数带回值,只规定函数完毕一定操作。,函数表达式,函数出目前一种表达式中,这种表达式称为函数表达式。这时规定函数带回一种确定值以参与表达式运算。例如:*(,);,第33页,函数参数,函数调用作为一种函数实参。例如:,m=max(a,max(b,c);,其中max(b,c)是一次函数调用,它值作为max另一次调用实参。m值是a、b、c三者中最大者。又如:printf(%d,max(a,b);也是把max(a,b)作为printf函数一种参数。,函数调用作为函数参数,实质上也是函数表达式形式调用一种,由于函数参数本来就规定是表达式形式。,第34页,8.对被调用函数申明和函数原型,在一种函数中调用另一函数(即被调用函数)需要具有哪些条件呢?,(1)首先被调用函数必须是已经存在函数(是库函数或顾客自己定义函数)。但光有这一条件还不够。,8.对被调用函数申明和函数原型,第35页,(3)假如使用顾客自己定义函数,而该函数位置在调用它函数(即主调函数)背面(在同一种文献中),应当在主调函数中对被调用函数作申明。,(2)假如使用库函数,还应当在本文献开头用命令将调用有关库函数时所需用到信息“包括”到本文献中来。,第36页,函数原型一般形式为,(1)函数类型 函数名(参数类型1,参数类型2);,(2)函数类型 函数名(参数类型1,参数名1,参数类型2,参数名2);,第37页,“申明”一词原文是delaration,过去在许多书中把它译为“阐明”。申明作用是把函数名、函数参数个数和参数类型等信息告知编译系统,以便在碰到函数调用时,编译系统能对旳识别函数并检查调用与否合法。(例如函数名与否对旳,实参与形参类型和个数与否一致)。,第38页,注意:函数“定义”和“申明”不是一回事。函数定义是指对函数功能确实立,包括指定函数名,函数值类型、形参及其类型、函数体等,它是一种完整、独立函数单位。而函数申明作用则是把函数名字、函数类型以及形参类型、个数和次序告知编译系统,以便在调用该函数时系统按此进行对照检查。,第39页,#include,void,main,(),float add(float x,float y);,*对被调用函数add申明*,float a,b,c;,scanf(f,f,a,b);,cadd(a,b);,printf(sum is f n,c);,float,add,(float,float),*函数首部*,float;,/*函数体*/,z;,return(z);,例8 对被调用函数作申明,第40页,假如 被调用函数定义出目前主调函数之前,可以不必加以申明。由于编译系统已经先懂得了已定义函数有关状况,会根据函数首部提供信息对函数调用作对旳性检查。,假如 被调用函数定义出目前主调函数之前,可以不必加以申明。由于编译系统已经先懂得了已定义函数有关状况,会根据函数首部提供信息对函数调用作对旳性检查。,第41页,改写例 8.,#include,float,add,(float,float),*函数首部*,float;,/*函数体*/,z;,return(z);,void,main,(),float a,b,c;,scanf(f,f,a,b);,cadd(a,b);,printf(sum is f n,c);,第42页,8.函数嵌套调用,嵌套定义就是在定义一种函数时,其函数体内又包括另一种函数完整定义。,语言不能嵌套定义函数,但可以嵌套调用函数,也就是说,在调用一种函数过程中,又调用另一种函数。,第43页,第44页,例 8.用弦截法求方程,f(x)=x3-5x2+16x-80=0 根,第45页,(1)取两个不一样样点x1,x2,假如f(x1)和f(x2)符号相反,则(x1,x2)区间内必有一种根。假如f(x1)与f(x2)同符号,则应变化x1,x2,直到f(x1)、f(x2)异号为止。注意x1、x2值不应差太大,以保证(x1,x2)区间内只有一种根。,(2)连接(x1,f(x1)和(x2,f(x2)两点,此线(即弦)交x轴于x。,措施:,第46页,(3)若f(x)与f(x1)同符号,则根必在(x,x2)区间内,此时将x作为新x1。假如f(x)与f(x2)同符号,则表达根在(x1,x)区间内,将x作为新x2。,(4)反复环节(2)和(3),直到 f(x)为止,为一种很小数,例如 10-6.此时认为 f(x)0,第47页,N-S流程图,第48页,分别用几种函数来实现各部分功能:,(1),用函数f(x)代表x函数:x3-5x2+16x-80.,(2),用函数调用xpoint(x1,x2)来求(x1,f(x1)和,(x2,f(x2)连线与x轴交点x坐标。,(3),用函数调用root(x1,x2)来求(x1,x2)区间,那个实根。显然,执行root函数过程中要用,到函数xpoint,而执行xpoint函数过程中要用,到f函数。,第49页,float f(float x),*定义函数,以实,现f(x)x3-5x2+16x-80*,;,=(-.)*+.)*-.;,();,第50页,float xpoint(float x1,float x2),*定义xpoint函数,求出弦与x轴交点*/,;,=(*()-*(),()-();,();,第51页,float root(float,float),/*定义root函数,求近似根*/,,;,();,do,(,);,();,if(*),/*()与()同符号*/,;,;,;,while,().);,();,第52页,void,main(),主函数,float,;,do,printf(,:);,scanf(,);,();,();,while,(*);,(,);,printf(root of equation is .n,);,运行状况如下:,,:,,,root of equation is,第53页,8函数递归调用,在调用一种函数过程中又出现直接或间接地调用该函数自身,称为函数递归调用。语言特点之一就在于容许函数递归调用。例如:,(int),,;,();,return(*);,第54页,第55页,例 8 有个人坐在一起,问第个人多少岁?他说比第个人大岁。问第个人岁数,他说比第个人大岁。问第个人,又说比第个人大岁。问第个人,说比第个人大岁。最终问第个人,他说是岁。请问第个人多大。,()(),()(),()(),()(),(),可以用数学公式表述如下:,()(),()(),第56页,可以用一种函数来描述上述递归过程:,int age(int)*求年龄递归函数*,int;*用作寄存函数返回值变量*,if();,else ();,return();,用一种主函数调用age函数,求得第5人年龄。,#include,void main(),(,();,运行成果如下:,第57页,例8.用递归措施求!,求!也可以用递归措施,即!等于!,而!。可用下面递归公式表达:,!(,),()!(),第58页,例 8.(汉诺)塔问题。这是一种古典数学问题,是一种用递归措施解题经典例子。问题是这样:古代有一种梵塔,塔内有3个座A、B、C,开始时座上有个盘子,盘子大小不等,大在下,小在上(见图8)。有一种老和尚想把这个盘子从座移到座,但每次只容许移动一种盘,且在移动过程中在3个座上都一直保持大盘在下,小盘在上。在移动过程中可以运用座,规定编程序打印出移动环节。,第59页,第60页,为便于理解,我们先分析将座上个盘子移到座上过程:,(1)将座上个盘子移到座上(借助);,(2)将座上个盘子移到座上;,(3)将座上个盘子移到座上(借助)。,其中第()步可以直接实现。第步又可用递归措施分解为:,将上个盘子从移到;,将上个盘子从移到;,将上个盘子从移到。,第()步可以分解为:,将上个盘子从移到上;,将上个盘子从移到上;,将上个盘子从移到上。,将以上综合起来,可得到移动3个盘子环节为,,。,第61页,由上面分析可知:将个盘子从座移到座可以分解为如下3个环节:,(1)将上个盘借助座先移到座上。,(2)把座上剩余一种盘移到座上。,(3)将个盘从座借助于座移到座上。,第62页,程序如下:,#include,void main(),void hanoi(int n,char one,char two,char three);,/*对hanoi函数申明*/,int m;,printf(input the number of diskes:);,scanf(“%d”,printf(The step to moveing%d diskes:n,m);,hanoi(m,A,B,C);,第63页,void,hanoi,(int n,char one,char two,char three),/*定义hanoi函数,将个盘从one座借助two座,移到three座*/,void,move,(char x,char y);,/*对move函数申明*/,if(n=1)move(one,three);,else,hanoi(n-1,one,three,two);,move(one,three);,hanoi(n-1,two,one,three);,void,move,(char x,char y),/*定义move函数*/,printf(“%c-%cn,x,y);,第64页,运行状况如下:,input the number of diskes:3,The steps to noving 3 diskes:,第65页,8.数组作为函数参数,8.7.1 数组元素作函数实参,由于实参可以是表达式,而数组元素可以是表达式构成部分,因此数组元素当然可以作为函数实参,与用变量作实参同样,是单向传递,即“值传送”方式。,例8.有两个数组和,各有个元素,将它们对应地逐一相比(即与比,与比)。假如数组中元素不小于数组中对应元素数目多于b数组中元素不小于a数组中对应元素数目(例如,aibi6次,biai3次,其中i每次为不一样样值),则认为a数组不小于b数组,并分别记录出两个数组对应元素不小于、等于、不不小于次数。,第66页,#include,void,main,(),int,large,(int x,int y);,/*函数申明*/,int 10,10,,;,printf(enter array a);,for(;),scanf(,);,printf();,printf(enter array);,for(;),scanf,(,);,printf();,for(;),if(,large,(i,i)=,);,else if(,large,(i,i)=)=+;,else;,第67页,printf(aibi%d timesnai=bi%d,timesnaik)printf(array a is larger than array bn);,else if(nk)printf(array a is smaller than array bn);,else printf(array is equal to array bn);,large,(int,int),int;,if();,else if()flag;,else flag;,return(flag);,第68页,运行状况如下:,enter array a:,5 3 8 9 1 3 5 6 0 4,array a is smaller thann array b,第69页,8.7.2 数组名作函数参数,可以用数组名作函数参数,此时形参应当用数组名或用指针变量。,例8.11 有一种一维数组,内放个学生成绩,求平均成绩。,第70页,#include,void,main,(),float,average,(float array10);,/*函数申明*/,float score10,aver;,int;,printf(input scores:);,for(;,scanf(,score);,printf();,aver,average,(score,);,printf,(average score is.n,aver);,第71页,float,average,(float array10),int;,float aver,;,for (;),;,;,return(aver);,运行状况如下:,input scores:,.5 .5 ,average score is 83.40,第72页,例 8.形参数组不定义长度,#include,void,main,(),float,average,(float ,int),float score_15,,.,;,float score_210=67.5,89.5,99,6.5,,77,89.5,76.5,54,60,99.5;,printf(“the average of class A is%6.2fn”,,average(score_1,5);,printf(“the average of class B is%6.2fn”,,average(score_2,10);,第73页,float,average,(float ,int),int;,float aver,;,for(;,sumsumarray;,aversum;,return();,运行成果如下:,the average of class A is 80.40,The average of class is 78.20,第74页,例 8.13 用选择法对数组中10个整数按由小到大排序。所谓选择法就是先将10个数中最小数与a0对换;再将a1到a9中最小数与a1对换每比较一轮,找出一种未经排序数中最小一种。共比较9轮。,第75页,a0 a1 a2 a3 a4,3 6 1 9 4 未排序时状况,1 6 3 9 4,将5个数中最小数1与a0对换,1 3 6 9 4,将余下4个数中最小数3与a1对换,1 3 4 9 6,将余下3个数中最小数4与a2对换,1 3 4 6 9,将余下2个数中最小数6与a3对,换,至此完毕排序,第76页,程序实例,#include,void,main,(),void,sort,(int,int);,int,;,printf(enter the array);,for(;,scanf(,);,sort,(,);,printf(the sorted array);,for(;,printf(,);,printf();,第77页,void,sort,(int,int),int,;,for(;),;,for(;),if(array array=;,=arrayk;,arrayk=arrayi;arrayi=t,第78页,8.7.3.多维数组名作函数参数,程序如下:,#include,void main(),max_value(int 4);,int 34=1,3,5,7,2,4,6,8,15,17,34,12;,printf(max value is,,max_value(a));,第79页,max_value,(int array 4),int,;,=;,(=;),(;,(array),=array,;,return();,运行成果如下:,Max value is 34,第80页,8.8局部变量和全局变量,8.8.1局部变量,在一种函数内部定义变量是内部变量,它只在本函数范围内有效,也就是说只有在本函数内才能使用它们,在此函数以外是不能使用这些变量。这称为“局部变量”。,第81页,float f1(int a),/*函数f1*/,int b,c;,a、b、c有效,char f2(int x,int y),/*函数f2*/,int i,j;,x、y、i、j有效,void main(),/*主函数*/,int m,n;,m、n有效,第82页,(1)主函数中定义变量(m,n)也只在主函数中有效,而不由于在主函数中定义而在整个文献或程序中有效。主函数也不能使用其他函数中定义变量。,(2)不一样样函数中可以使用相似名字变量,它们代表不一样样对象,互不干扰。例如,上面在f1函数中定义了变量b和c,倘若在f2函数中也定义变量b和c,它们在内存中占不一样样单元,互不混淆。,(3)形式参数也是局部变量。例如上面f1函数中形参a,也只在f1函数中有效。其他函数可以调用f1函数,但不能引用f1函数形参a。,(4)在一种函数内部,可以在复合语句中定义变量,这些变量只在本复合语句中有效,这种复合语句也称为“分程序”或“程序块”。,说,明,第83页,void main(),int a,b;,int c;,c=a+b;,c在此范围内有效 a,b在此范围内有效,第84页,8.8.2 全局变量,在函数内定义变量是局部变量,而在函数之外定义变量称为外部变量,外部变量是全局变量(也称全程变量)。全局变量可认为本文献中其他函数所共用。它有效范围为从定义变量位置开始到本源文献结束。,第85页,int p=1,q=5;,/*外部变量*/,float f1(int a),/*定义函数f1*/,int b,c;,char c1,c2;,/*外部变量*/,char f2(int x,int y),/*定义函数f2*/,int i,j;,全局变量p,q作用范围,全局变量c1,c2作用范围,void main(),/*主函数*/,int m,n;,第86页,例 8.15 有一种一维数组,内放个学生成绩,写一种函数,求出平均分、最高分和最低分。,#include,float Max,Min;,*全局变量*,void,main,(),float,average,(float array,int n);,float ave,score10;,int;,for(;),scanf(,);,ave=,average,(,);,printf(“max=%6.2fnmin=%6.2fn,average=%6.2fn“,Max,Min,ave);,第87页,float,average,(float array,int n),*定义函数,形参为数组*/,int;,float aver,sum=array;,Max=Min=array;,for(=;),if(arrayMax)Maxarray;,else if(arrayMin)Min array;,sum=sum+array;,aver;,return();,运行状况如下:,第88页,第89页,提议不在必要时不要使用全局变量,原因如下:,全局变量在程序所有执行过程中都占用寄存单元,而不是仅在需要时才开辟单元。,使用全局变量过多,会减少程序清晰性,人们往往难以清晰地判断出每个瞬时各个外部变量值。在各个函数执行时都也许变化外部变量值,程序轻易出错。因此,要限制使用全局变量。,第90页,它使函数通用性减少了,由于函数在执行时要依赖于其所在外部变量。假如将一种函数移到另一种文献中,还要将有关外部变量及其值一起移过去。但若该外部变量与其他文献变量同名时,就会出现问题,减少了程序可靠性和通用性。一般规定把程序中函数做成一种封闭体,除了可以通过“实参形参”渠道与外界发生联络外,没有其他渠道。,第91页,例 8.6 外部变量与局部变量同名,#include,int a=3,b=5;,/*a,b为外部变量*/a,b作用范围,void,main,(),int a=8;,/*a为局部变量*/局部变量a作用范围,printf(%d,max,(a,b);,全局变量b作用范围,max,(int a,int b),/*a,b为局部变量*/,int c;,c=ab?ab;,形参a、b作用范围,return(c);,运行成果为,8,第92页,8.变量寄存类别,8.动态寄存方式与静态寄存方式,前面已简介了从变量作用域(即从空间)角度来分,可以分为全局变量和局部变量。那么从变量值存在时间(即生存期)角度来分,又可以分为静态寄存方式和动态寄存方式。,所谓静态寄存方式是指在程序运行期间由系统分派固定寄存空间方式。而动态寄存方式则是在程序运行期间根据需要进行动态分派寄存空间方式。这个寄存空间可以分为三部分,,1.程序区,2.静态寄存区,3.动态寄存区,第93页,在语言中每一种变量和函数有两个属性:数据类型和数据寄存类别。对数据类型,读者已熟悉(如整型、字符型等)。寄存类别指是数据在内存中寄存方式。寄存方式分为两大类:静态寄存类和动态寄存类。详细包括四种:自动(),静态(),寄存器(),外部()。根据变量寄存类别,可以懂得变量作用域和生存期。,第94页,8.auto变量,函数中局部变量,如不专门申明为static寄存类别,都是动态地分派寄存空间,数据寄存在动态寄存区中。函数中形参和在函数中定义变量(包括在复合语句中定义变量),都属此类,在调用该函数时系统会给它们分派寄存空间,在函数调用结束时就自动释放这些寄存空间。因此此类局部变量称为自动变量。自动变量用关键字作寄存类别申明。例如:,int(int)*定义f函数,为形参*,auto int,;*定义、为自动变量*,第95页,8.9.3用static申明局部变量,有时但愿函数中局部变量值在函数调用结束后不消失而保留原值,即其占用寄存单元不释放,在下一次该函数调用时,该变量已经有值,就是上一次函数调用结束时值。这时就应当指定该局部变量为“静态局部变量”,用关键字进行申明。通过下面简朴例子可以理解它特点。,第96页,例87 考察静态局部变量值。,#include,void main(),int(int);,,;,(;,(,();,int(int),int;,;,;,;,();,第97页,对静态局部变量阐明:,(1)静态局部变量属于静态寄存类别,在静态寄存区内分派寄存单元。在程序整个运行期间都不释放。而自动变量(即动态局部变量)属于动态寄存类别,占动态寄存区空间而不占静态寄存区空间,函数调用结束后即释放。,(2)对静态局部变量是在编译时赋初值,即只赋初值一次,在程序运行时它已经有初值。后来每次调用函数时不再重新赋初值而只是保留上次函数调用结束时值。而对自动变量赋初值,不是在编译时进行,而是在函数调用时进行,每调用一次函数重新给一次初值,相称于执行一次赋值语句。,第98页,(3)如在定义局部变量时不赋初值话,则对静态局部变量来说,编译时自动赋初值(对数值型变量)或空字符(对字符变量)。而对自动变量来说,假如不赋初值则它值是一种不确定值。这是由于每次函数调用结束后寄存单元已释放,下次调用时又重新另分派寄存单元,而所分派单元中值是不确定。,(4)虽然静态局部变量在函数调用结束后仍然存在,但其他函数是不能引用它。,第99页,例88,输出到阶乘值。,#include,void,main,(),int fac(int);,int;,for(;),printf(%!=,fac();,Int fac(int),static int;,*;,();,第100页,8.9.4 register变量,一般状况下,变量(包括静态寄存方式和动态寄存方式)值是寄存在内存中。当程序中用到哪一种变量值时,由控制器发出指令将内存中该变量值送到运算器中。通过运算器进行运算,假如需要存数,再从运算器将数据送到内存寄存。,第101页,假如有某些变量使用频繁(例如在一种函数中执行次循环,每次循环中都要引用某局部变量),则为存取变量值要花费不少时间。为提高执行效率,语言容许将局部变量值放在CPU中寄存器中,需要用时直接从寄存器取出参与运算,不必再到内存中去存取。由于对寄存器存取速度远高于对内存存取速度,因此这样做可以提高执行效率。这种变量叫做寄存器变量,用关键字作申明。例如,例819中程序是输出到n阶乘值。,第102页,例819使用寄存器变量,#include,void main(),long fac(long);,long i,n;,scanf(%ld,for(i=1;i=n;i+),printf(%ld!=%ldn,i,fac(i);,long fac(long n),register long i,f=1;/*定义寄存器变量*/,for(i=1;iy?x:y;,return(z);,第105页,2.在多文献程序中申明外部变量,例8 用extern将外部变量作用域扩展到其他文献。本程序作用是给定值,输入和,,求和am值。文献file中内容为:,#include,int A;/*定义外部变量*/,void main(),int(int);/*函数申明*/,int,;,printf(enter the number a and its power m:n);,scanf(,A,);,A*;,printf(*,A,);,();,printf(*n,A,);,第106页,文献file中内容为:,extern A;/*申明A为一种已定义外部变量*/,int(int);,int,;,for(;),*A;,();,第107页,8.9.6用static申明外部变量,有时在程序设计中但愿某些外部变量只限于被本文献引用,而不能被其他文献引用。这时可以在定义外部变量时加一种申明。,例如:,file1.c file2.c,static int A;extern int A;,void main()void fun(int n),A=A*n;,第108页,8.9.7有关变量申明和定义,对变量而言,申明与定义关系稍微复杂某些。在申明部分出现变量有两种状况:一种是需要建立寄存空间(如:int a;),另一种是不需要建立寄存空间(如:extern a;)。前者称为“定义性申明”(defining declaration),或简称定义(definition)。后者称为“引用性申明”(referencing d
展开阅读全文

开通  VIP会员、SVIP会员  优惠大
下载10份以上建议开通VIP会员
下载20份以上建议开通SVIP会员


开通VIP      成为共赢上传
相似文档                                   自信AI助手自信AI助手

当前位置:首页 > 包罗万象 > 大杂烩

移动网页_全站_页脚广告1

关于我们      便捷服务       自信AI       AI导航        抽奖活动

©2010-2025 宁波自信网络信息技术有限公司  版权所有

客服电话:4009-655-100  投诉/维权电话:18658249818

gongan.png浙公网安备33021202000488号   

icp.png浙ICP备2021020529号-1  |  浙B2-20240490  

关注我们 :微信公众号    抖音    微博    LOFTER 

客服