第7章 用函数实现模块化程序设计_第1页
第7章 用函数实现模块化程序设计_第2页
第7章 用函数实现模块化程序设计_第3页
第7章 用函数实现模块化程序设计_第4页
第7章 用函数实现模块化程序设计_第5页
已阅读5页,还剩103页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第7章用函数实现模块化程序设计7.1为什么要用函数

7.2怎样定义函数7.3调用函数7.4对被调用函数的声明和函数原型7.5函数的嵌套调用

7.6函数的递归调用7.7数组作为函数参数

7.8局部变量和全局变量7.9变量的存储方式和生存期7.10关于变量的声明和定义7.11内部函数和外部函数7.1为什么要用函数问题:如果程序的功能比较多,规模比较大,把所有代码都写在main函数中,就会使主函数变得庞杂、头绪不清,阅读和维护变得困难有时程序中要多次实现某一功能,就需要多次重复编写实现此功能的程序代码,这使程序冗长,不精炼7.1为什么要用函数解决的方法:用模块化程序设计的思路采用“组装”的办法简化程序设计的过程事先编好一批实现各种不同功能的函数把它们保存在函数库中,需要时直接用7.1为什么要用函数解决的方法:用模块化程序设计的思路函数就是功能每一个函数用来实现一个特定的功能函数的名字应反映其代表的功能7.1为什么要用函数在设计一个较大的程序时,往往把它分为若干个程序模块,每一个模块包括一个或多个函数,每个函数实现一个特定的功能C程序可由一个主函数和若干个其他函数构成主函数调用其他函数,其他函数也可以互相调用同一个函数可以被一个或多个函数调用任意多次7.1为什么要用函数一个较大的程序可分为若干个程序模块,每一个模块用来实现一个特定的功能。在高级语言中用子程序实现模块的功能。子程序由函数来完成。一个C程序可由一个主函数和若干个其他函数构成。由主函数调用其他函数,其他函数也可以互相调用。同一个函数可以被一个或多个函数调用任意多次。函数间的调用关系7.1为什么要用函数可以使用库函数可以使用自己编写的函数在程序设计中要善于利用函数,可以减少重复编写程序段的工作量,同时可以方便地实现模块化的程序设计7.1为什么要用函数例7.1输出以下的结果,用函数调用实现。******************Howdoyoudo!****************** main() { PrintStar(); PrintMessage(); PrintStar(); } PrintStar() { printf(“**********\n”); } PrintMessage() { printf(“Hello\n”); }PrintStar函数PrintMessage函数main函数主函数两个自定义函数**********Hello**********7.2怎样定义函数7.2.1为什么要定义函数7.2.2定义函数的方法7.2.1为什么要定义函数C语言要求,在程序中用到的所有函数,必须“先定义,后使用”指定函数名字、函数返回值类型、函数实现的功能以及参数的个数与类型,将这些信息通知编译系统。7.2.1为什么要定义函数指定函数的名字,以便以后按名调用指定函数类型,即函数返回值的类型指定函数参数的名字和类型,以便在调用函数时向它们传递数据指定函数的功能。这是最重要的,这是在函数体中解决的7.2.1为什么要定义函数对于库函数,程序设计者只需用#include指令把有关的头文件包含到本文件模块中即可程序设计者需要在程序中自己定义想用的而库函数并没有提供的函数7.2.2定义函数的方法函数定义的一般形式 类型符函数名(形式参数说明表) { 说明部分 语句部分

}函数首部函数体intadd(intx,inty){ intsum; sum=x+y; returnsum;}完整的函数定义无参函数定义空函数定义intread(){ intx; scanf(“%d”,&x); returnx;}

dummy(){};[例7.2]实参和形参voidmain(){ inta=15,b=20,c; c=max(a+10,b); printf(“Max=%d”,c);}intmax(intx,inty){ intz; z=x>y?x:y; returnz;}main函数max函数形参实参15a20bczxy252025257.3调用函数7.3.1函数调用的形式7.3.2函数调用时的数据传递7.3.3函数调用的过程7.3.4函数的返回值7.3.1函数调用的形式函数调用的一般形式函数名(实参表)如果是调用无参函数,则实参表可以没有,但括号不能省略。如果实参表包含多个实参,则各参数之间用逗号隔开。实参与形参的个数应相等,类型应一致。实参与形参按顺序对应,一一传递数据。如:

p=power(2.5,4); /*ok*/ q=power(2.5,3.8); /*error*/ s=power(2.5); /*error*/ t=power(5,3); /*ok*/函数调用的方式1.函数语句:即把函数调用作为一个语句。如:

printStar(); printf(“***********”);

这时不要求函数带回值,只要求完成一定的操作,通常将函数字义为“void”类型。2.函数表达式:函数出现在可以写表达式的地方,此时需要的是函数的值。如:c=2*max(a,b);printf(“%d”,max(a,b));7.3.2函数调用时的数据传递1.形式参数和实际参数在调用有参函数时,主调函数和被调用函数之间有数据传递关系定义函数时函数名后面的变量名称为“形式参数”(简称“形参”)主调函数中调用一个函数时,函数名后面参数称为“实际参数”(简称“实参”)实际参数可以是常量、变量或表达式7.3.2函数调用时的数据传递2.实参和形参间的数据传递在调用函数过程中,系统会把实参的值传递给被调用函数的形参或者说,形参从实参得到一个值该值在函数调用期间有效,可以参加被调函数中的运算7.3.2函数调用时的数据传递

例7.3输入两个整数,要求输出其中值较大者。要求用函数来找到大数。先编写max函数:intmax(intx,inty){intz;z=x>y?x:y;return(z);}7.3.2函数调用时的数据传递在max函数上面,再编写主函数#include<stdio.h>intmain(){intmax(intx,inty);inta,b,c;printf(“twointegernumbers:");scanf(“%d,%d”,&a,&b);c=max(a,b);printf(“maxis%d\n”,c);}实参可以是常量、变量或表达式7.3.2函数调用时的数据传递

c=max(a,b);(main函数)intmax(intx,inty)(max函数){intz;z=x>y?x:y;return(z);}7.3.3函数调用的过程在定义函数中指定的形参,在未出现函数调用时,它们并不占内存中的存储单元。在发生函数调用时,函数max的形参被临时分配内存单元。2a3bxy23实参形参7.3.3函数调用的过程调用结束,形参单元被释放实参单元仍保留并维持原值,没有改变如果在执行一个被调用函数时,形参的值发生改变,不会改变主调函数的实参的值2a3bxy23实参形参7.3.4.函数的返回值通常,在调用一个函数时,都希望被调函数能返回一个确定的值。这就是函数的返回值。函数的返回值是通过返回语句return实现的,返回语句的形式为:

return;或return表达式;前一种形式,函数返回值是不确定的,这时调用者一般对返回值不感兴趣,同时将函数说明成void类型。后一种形式,函数要把“表达式”的值返回给调用者。7.4对被调用函数的声明和函数原型在一个函数中调用另一个函数需要具备如下条件:(1)被调用函数必须是已经定义的函数(是库函数或用户自己定义的函数)(2)如果使用库函数,应该在本文件开头加相应的#include指令(3)如果使用自己定义的函数,而该函数的位置在调用它的函数后面,应该声明对被调函数的声明

voidmain() {inta=10,b=20,c;c=max(a,b); printf(“%d\n”,c); } intmax(intx,inty) {returnx>y?x:y; }怎样使这些信息正确呢?C编译系统假定:

1、函数名:max2、函数类型:int根据扫描得知:

3、有两个参数

4、每一个参数的类型均为int该函数与已知信息相符。floatfloatfloatfloat%f由于类型的变化,使该函数与已知信息不相符。这句有错误!float函数声明有三种格式:格式1:函数类型函数名(形参说明表);格式2:函数类型函数名(形参类型表);格式3:函数类型函数名();如:intmax(intx,inty);intmax(int,int);intmax();例7.4求最大值函数原型#include<stdio.h>floatmax(floatx,floaty){returnx>y?x:y;}main(){ floata=10,b=20,c; c=max(a,b); printf("%f\n",c);}①②④⑥⑦保存:返回地址当前现场③恢复:主调程序现场返回地址⑤main()调fun()结束fun()返回图7.1函数调用的过程练习:1.计算n!的数学定义为: 请编写函数fact()求n!。2.在上题基础上编写函数求。m与n通过键盘输入,注意判断m与n的关系。7.5函数的嵌套调用C语言不能嵌套定义函数,但可以嵌套调用函数,也就是说,在调用一个函数的过程中,又调用另一个函数。7.5函数的嵌套调用

例7.5输入4个整数,找出其中最大的数。用函数的嵌套调用来处理。解题思路:main中调用max4函数,找4个数中最大者max4中再调用max2,找两个数中的大者max4中多次调用max2,可找4个数中的大者,然后把它作为函数值返回main函数main函数中输出结果#include<stdio.h>intmain(){intmax4(inta,intb,intc,intd);inta,b,c,d,max;printf(“4intergernumbers:");scanf("%d%d%d%d",&a,&b,&c,&d);max=max4(a,b,c,d);printf("max=%d\n",max);return0;}主函数对max4

函数声明输入4个整数调用后肯定是4个数中最大者输出最大者intmax4(inta,intb,intc,intd){intmax2(inta,intb);intm;m=max2(a,b);m=max2(m,c);m=max2(m,d);return(m);}max4函数对max2

函数声明a,b中较大者a,b,c中较大者a,b,c,d中最大者intmax2(inta,intb){if(a>=b)returna;elsereturnb;}找a,b中较大者return(a>b?a:b);intmax4(inta,intb,intc,intd){intmax2(inta,intb);ruturnmax2(max2(max2(a,b),c),d);}intmax2(inta,intb){return(a>b?a:b);}#include<stdio.h>intmain(){……max=max4(a,b,c,d);……}7.6函数的递归调用在调用一个函数的过程中又出现直接或间接地调用该函数本身,称为函数的递归调用。C语言的特点之一就在于允许函数的递归调用。7.6函数的递归调用

例7.6有5个学生坐在一起问第5个学生多少岁?他说比第4个学生大2岁问第4个学生岁数,他说比第3个学生大2岁问第3个学生,又说比第2个学生大2岁问第2个学生,说比第1个学生大2岁最后问第1个学生,他说是10岁请问第5个学生多大age(5)=age(4)+2age(4)=age(3)+2age(3)=age(2)+2age(2)=age(1)+2age(1)=10age(2)=12age(3)=14age(4)=16age(5)=18

回溯阶段

递推阶段结束递归的条件#include<stdio.h>intmain(){intage(intn);printf("NO.5,age:%d\n",age(5));return0;}

intage(intn){intc;if(n==1)c=10;elsec=age(n-1)+2;return(c);}age(5)输出age(5)mainc=age(4)+2age函数n=5c=age(3)+2age函数n=4c=age(1)+2age函数n=2c=age(2)+2age函数n=3c=10age函数n=1age(1)=10age(2)=12age(3)=14age(4)=16age(5)=1818例7.7用递归方法求n!。#include<stdio.h>intmain(){intfac(intn);intn;inty;printf("inputanintegernumber:");scanf("%d",&n);y=fac(n);printf("%d!=%d\n",n,y);return0;}intfac(intn){intf;if(n<0) printf("n<0,dataerror!");elseif(n==0||n==1) f=1;elsef=fac(n-1)*n;return(f);}注意溢出fac(5)输出fac(5)mainf=fac(4)×5fac函数n=5f=fac(3)×4fac函数n=4f=fac(1)×2fac函数n=2f=fac(2)×3fac函数n=3f=1fac函数n=1fac(1)=1fac(2)=2fac(3)=6fac(4)=24fac(5)=120120

例7.8Hanoi(汉诺)塔问题。古代有一个梵塔,塔内有3个座A、B、C,开始时A座上有64个盘子,盘子大小不等,大的在下,小的在上。有一个老和尚想把这64个盘子从A座移到C座,但规定每次只允许移动一个盘,且在移动过程中在3个座上都始终保持大盘在下,小盘在上。在移动过程中可以利用B座。要求编程序输出移动一盘子的步骤。ABCABC……将63个从A到B第1个和尚的做法……ABC将63个从A到B第1个和尚的做法……ABC将1个从A到C第1个和尚的做法……ABC将1个从A到C第1个和尚的做法……ABC将63个从B到C第1个和尚的做法……ABC将63个从B到C第1个和尚的做法ABC……将62个从A到C第2个和尚的做法ABC……将62个从A到C第2个和尚的做法ABC……将1个从A到B第2个和尚的做法ABC……将1个从A到B第2个和尚的做法ABC……将62个从C到B第2个和尚的做法ABC……将62个从C到B第2个和尚的做法第3个和尚的做法第4个和尚的做法第5个和尚的做法第6个和尚的做法第7个和尚的做法……第63个和尚的做法第64个和尚仅做:将1个从A移到CABC将3个盘子从A移到C的全过程将2个盘子从A移到BABC将3个盘子从A移到C的全过程将2个盘子从A移到BABC将3个盘子从A移到C的全过程将1个盘子从A移到CABC将3个盘子从A移到C的全过程将1个盘子从A移到CABC将3个盘子从A移到C的全过程将2个盘子从B移到CABC将3个盘子从A移到C的全过程将2个盘子从B移到CABC将2个盘子从A移到B的过程将1个盘子从A移到CABC将2个盘子从A移到B的过程将1个盘子从A移到CABC将2个盘子从A移到B的过程将1个盘子从A移到BABC将2个盘子从A移到B的过程将1个盘子从A移到BABC将2个盘子从A移到B的过程将1个盘子从C移到BABC将2个盘子从A移到B的过程将1个盘子从C移到BABC将2个盘子从B移到C的过程ABC将2个盘子从B移到C的过程ABC将2个盘子从B移到C的过程ABC将2个盘子从B移到C的过程由上面的分析可知:将n个盘子从A座移到C座可以分解为以下3个步骤:(1)将A上n-1个盘借助C座先移到B座上(2)把A座上剩下的一个盘移到C座上(3)将n-1个盘从B座借助于A座移到C座上#include<stdio.h>intmain(){voidhanoi(intn,charone,chartwo,charthree);intm;printf(“thenumberofdiskes:");scanf("%d",&m);printf("move%ddiskes:\n",m);

hanoi(m,'A','B','C');}voidhanoi(intn,charone,chartwo,charthree){voidmove(charx,chary);if(n==1)

move(one,three);else{hanoi(n-1,one,three,two);

move(one,three);

hanoi(n-1,two,one,three);}}voidmove(charx,chary){printf("%c-->%c\n",x,y);}7.7数组作为函数参数7.7.1数组元素作函数实参7.7.2数组名作函数参数7.7.3多维数组名作函数参数7.7.1数组元素作函数实参

例7.8输入10个数,要求输出其中值最大的元素和该数是第几个数。#include<stdio.h>intmain(){intmax(intx,inty);inta[10],m,n,i;printf(“10integernumbers:\n");for(i=0;i<10;i++)scanf("%d",&a[i]);printf("\n");for(i=1,m=a[0],n=0;i<10;i++){if(max(m,a[i])>m) {m=max(m,a[i]); n=i; }}printf(“largestnumberis%d\n",m);printf(“%dthnumber.\n“,n+1);}intmax(intx,inty){return(x>y?x:y);}7.7.2数组名作函数参数除了可以用数组元素作为函数参数外,还可以用数组名作函数参数(包括实参和形参)用数组元素作实参时,向形参变量传递的是数组元素的值用数组名作函数实参时,向形参

传递的是数组首元素的地址7.7.2数组名作函数参数

例7.9有一个一维数组score,内放10个学生成绩,求平均成绩。#include<stdio.h>intmain(){floataverage(floatarray[10]);

floatscore[10],aver;inti;printf("input10scores:\n");for(i=0;i<10;i++)scanf("%f",&score[i]);printf("\n");aver=average(score);printf("%5.2f\n",aver);return0;}定义实参数组floataverage(float

array[10]){inti;floataver,sum=array[0];for(i=1;i<10;i++)sum=sum+array[i];aver=sum/10;return(aver);}定义形参数组相当于score[0]相当于score[i]

例7.10用选择法对数组中10个整数按由小到大排序。a[0]a[1]a[2]a[3]a[4]36194

16394

1

3694

1

3

496

1

3

4

69小到大排序#include<stdio.h>intmain(){voidsort(intarray[],intn);inta[10],i;printf("enterarray:\n");for(i=0;i<10;i++)scanf("%d",&a[i]);

sort(a,10);printf("Thesortedarray:\n");for(i=0;i<10;i++)printf("%d",a[i]);printf("\n");return0;}voidsort(intarray[],intn){inti,j,k,t;for(i=0;i<n-1;i++){k=i;for(j=i+1;j<n;j++)if(array[j]<array[k])k=j; t=array[k];

array[k]=array[i];

array[i]=t; }}在sort[i]~sort[9]中,最小数与sort[i]对换7.7.3多维数组名作函数参数

例7.11有一个3×4的矩阵,求所有元素中的最大值。#include<stdio.h>intmain(){intmax_value(intarray[][4]);inta[3][4]={{1,3,5,7},{2,4,6,8},{15,17,34,12}};printf(“Maxvalueis%d\n”,

max_value(a));return0;}可以省略不能省略要与形参数组第二维大小相同intmax_value(intarray[][4]){inti,j,max;max=array[0][0];for(i=0;i<3;i++)for(j=0;j<4;j++)if(array[i][j]>max)max=array[i][j];return(max);}要与实参数组第二维大小相同7.8局部变量和全局变量7.8.1局部变量7.8.2全局变量7.8.1局部变量定义变量可能有三种情况:在函数的开头定义在函数内的复合语句内定义在函数的外部定义全局和局部变量charch;doublefun2(charc){ floatb;

…… ch=c;}intx;main(){ intm,n;

……}在函数内部说明的变量是局部变量。作用域:自说明位置起至函数结束止。在函数外部说明的变量是全局部变量。作用域:自说明位置起至文件结束止。cbm,nchx在定义一个变量时,除了指定其数据类型外,还可以指定其存储类别。C语言中用四个关键字来表示存储类别:auto————自动的register———寄存器的static ———静态的extern———外部的局部变量全局变量7.8.1局部变量自动变量(auto)intf(inta){ autointb,c=3; ……}自动变量在内存的动态存储区分配:当函数(或分程序)执行时为自动变量分配存储空间;当函数(或分程序)结束时释放自动变量所占的存储空间。auto关键字可以省略。寄存器变量(register)intf(inta){ registerinti; ……}寄存器变量存在于CPU的寄存器中:当函数(或分程序)执行时为寄存器变量分配寄存器;当函数(或分程序)结束时释放寄存器变量所占用的寄存器。自动变量与寄存器变量之间的区别自动变量main(){

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论