《第6章 函数与模块化程序设计基础PPT讲稿.ppt》由会员分享,可在线阅读,更多相关《第6章 函数与模块化程序设计基础PPT讲稿.ppt(45页珍藏版)》请在taowenge.com淘文阁网|工程机械CAD图纸|机械工程制图|CAD装配图下载|SolidWorks_CaTia_CAD_UG_PROE_设计图分享下载上搜索。
1、第第6章章 函数与模块化函数与模块化程序设计基础程序设计基础1第1页,共45页,编辑于2022年,星期一6.1 概述概述6.1.1 模块与函数模块与函数1.功能模块功能模块求解较小问题的算法和程序称作求解较小问题的算法和程序称作“功能模块功能模块”,各功能模块可以先单独设计,然后将求解各功能模块可以先单独设计,然后将求解所有子问题的模块组合成求解原问题的程序。所有子问题的模块组合成求解原问题的程序。“自顶向下自顶向下”的模块化程序设计方法:的模块化程序设计方法:将一个大问题分解成多个解决小问题的模块将一个大问题分解成多个解决小问题的模块的设计思想。的设计思想。2第2页,共45页,编辑于2022
2、年,星期一2.由功能模块组成程序的结构图由功能模块组成程序的结构图:主控模块主控模块模块模块1_1模块模块1_n模块模块2_1模块模块2_n模块模块n_1模块模块n_n模块模块1模块模块2模块模块n3.函数函数:完成相对独立功能的程序完成相对独立功能的程序3第3页,共45页,编辑于2022年,星期一【例【例6-1】输入年月日,计算出该日为该年的第几天。输入年月日,计算出该日为该年的第几天。主控模块主控模块判断闰年判断闰年求某月的天数求某月的天数输输 出出输输 入入求总天数求总天数图图6-2 6-2 程序结构图程序结构图4第4页,共45页,编辑于2022年,星期一程序实现程序实现:(1)判断闰年
3、。)判断闰年。int leap(int year)int lp;lp=(year%4=0&year%100!=0|year%400=0)?1:0;return lp;5第5页,共45页,编辑于2022年,星期一(2 2)求某月的天数。)求某月的天数。int month_days(int year,int month)int month_days(int year,int month)int ds,d;int ds,d;switch(month)switch(month)case 1:case 1:case 3:case 3:case 5:case 5:case 7:case 7:case 8:
4、case 8:case 10:case 10:case 12:d=31;break;case 12:d=31;break;case 2:d=leap(year)?29:28;break;case 2:d=leap(year)?29:28;break;default:d=30;default:d=30;return d;return d;6第6页,共45页,编辑于2022年,星期一(3 3)求天数和。)求天数和。int days(int year,int month,int day)int i,ds=0;for(i=1;imonth;i+)ds=ds+month_days(year,i);ds=
5、ds+day;return ds;7第7页,共45页,编辑于2022年,星期一4)在主函数中分别调用三个函数。在主函数中分别调用三个函数。void main()int year,month,day,t_day;printf(Input year-month-day:n);scanf(%d-%d-%d,&year,&month,&day);t_day=days(year,month,day);printf(%d-%d-%d is%dth day of the year!n,year,month,day,t_day);注意注意:在完整的程序中在完整的程序中,前三个函数应放在前三个函数应放在main
6、()函数之前。函数之前。8第8页,共45页,编辑于2022年,星期一6.1.2 模块设计三个原则模块设计三个原则模块独立。模块独立。功能独立的子功能功能独立的子功能模块之间的关系简单模块之间的关系简单使用独立变量使用独立变量模块规模适当模块规模适当分解模块要注意层次分解模块要注意层次对问题抽象化对问题抽象化设计时细化设计时细化9第9页,共45页,编辑于2022年,星期一6.2 函数定义与使用函数定义与使用一、标准库函数一、标准库函数定义在不同的头文件中定义在不同的头文件中用户使用时,必须用用户使用时,必须用#include“头文件头文件”把相应的头把相应的头文件包含到程序中来。文件包含到程序中
7、来。#include /*包含包含math.h头文件头文件 */#include /*包含包含 stdio.h 头文件头文件*/main()double a,b;scanf(“%f“,&a);/*调用输入函数,输入变量调用输入函数,输入变量a的值的值*/b=sin(a);/*调用调用sin函数,求函数,求sin(a)的值的值*/printf(“%6.4f”,b);/*调用输出函数,输出变量调用输出函数,输出变量b的值的值*/10第10页,共45页,编辑于2022年,星期一二、用户自定义函数二、用户自定义函数1.函数类型函数类型无参函数无参函数函数的定义无参数说明函数的定义无参数说明 有参函数有
8、参函数 定义的参数有一个或一个以上的参数定义的参数有一个或一个以上的参数 空函数空函数当定义的函数既无参数也无执行语句。当定义的函数既无参数也无执行语句。空函数被调用时,什么也不做立即返回其调用函数。空函数被调用时,什么也不做立即返回其调用函数。11第11页,共45页,编辑于2022年,星期一2.函数定义函数定义方式方式1 函数返回值类型名函数返回值类型名 函数名函数名(参数列表参数列表)参数类型说明参数类型说明 局部变量说明局部变量说明;语句序列语句序列;方式方式2 函数返回值类型名函数返回值类型名 函数名函数名(参数类型说明及参数列表参数类型说明及参数列表)局部变量说明局部变量说明;语句序
9、列语句序列;如如:int max(a,b)int a,b;如如:int max(int a,int b)12第12页,共45页,编辑于2022年,星期一【例【例6-3】定义符号函数】定义符号函数sign。int sign(x)/*函数返回值类型未说明,默认为函数返回值类型未说明,默认为int,建议给出函数类型说明,建议给出函数类型说明*/int x;/*形式参数说明形式参数说明*/int y;/*函数体局部变量函数体局部变量*/y=x0?1:(x=0?0:-1);return y;/*返回函数值返回函数值*/注意注意:C语言函数分为两大部分语言函数分为两大部分:函数的说明部分函数的说明部分函数
10、体部分。函数体部分。13第13页,共45页,编辑于2022年,星期一函数各部分作用函数各部分作用1)函数的说明部分函数的说明部分 函数说明部分说明函数的类型函数说明部分说明函数的类型,函数名函数名,参数表及参参数表及参数类型。数类型。(1)函数的类型说明函数的类型说明函数的类型即函数的返回值类型。若函数不提供返回值,则函数的类型即函数的返回值类型。若函数不提供返回值,则可定义其类型为可定义其类型为:void。例如例如:void putdata(int a)(2)函数名函数名 函数名又称函数标识符。命名遵循函数名又称函数标识符。命名遵循C语语言标识符的规定;语语言标识符的规定;函数名要反映函数完
11、成的功能。函数名要反映函数完成的功能。14第14页,共45页,编辑于2022年,星期一(3)参数表参数表参数表写在函数名后的参数表写在函数名后的()内,由一个或多个变量标内,由一个或多个变量标识符及类型标识符组成。识符及类型标识符组成。参数表中的变量称为形式参数参数表中的变量称为形式参数,简称形参。简称形参。若函数没有形参,则称为无参函数,其后若函数没有形参,则称为无参函数,其后“()”不能省略。不能省略。参数必须指定类型。形参的类型说明有两种:参数必须指定类型。形参的类型说明有两种:方法方法1:int max(a,b)int a,b;方法方法2:int max(int a,int b)省略函
12、数类型名时,省略函数类型名时,C语言默认其为语言默认其为int型。型。15第15页,共45页,编辑于2022年,星期一2)函数体函数体函数体包括变量定义和执行语句序列。函数所完成函数体包括变量定义和执行语句序列。函数所完成的工作由函数体中一段程序实现。的工作由函数体中一段程序实现。函数的返回值用返回语句函数的返回值用返回语句return返回,形式返回,形式:return(表达式表达式);或或 return 表达式;表达式;如果函数的类型与如果函数的类型与return语句的表达式的类语句的表达式的类型不一致时型不一致时,则以函数的类型为准。返回时则以函数的类型为准。返回时自动进行数据转换。自动进
13、行数据转换。(见下页例题见下页例题)16第16页,共45页,编辑于2022年,星期一例例6.3 定义函数定义函数power(x,n),求求x的的n次方。次方。函数定义如下函数定义如下:float power(float x,int n)int i;float t=1;for(i=1;ib)?a:b;y=yc?y:c;printf(max=%dn,y);void main()int x,y,z,m;scanf(%d,%d,%d,&x,&y,&z);max(x,y,z);/*采用函数语句形式调用函数采用函数语句形式调用函数max*/19第19页,共45页,编辑于2022年,星期一函数调用形式函数调
14、用形式int max(int a,int b)int y;y=(ab)?a:b;return y;void main()int x,y,z,m;scanf(%d,%d,%d,&x,&y,&z);m=max(x,y);m=max(m,z);printf(“max=%dn”,m);/*表达式调用形式表达式调用形式*/m=max(x,y);printf(max=%dn,max(m,z);/*函数参数调用形式函数参数调用形式*/20第20页,共45页,编辑于2022年,星期一2.函数声明函数声明函数定义在函数定义在main()之后,需要进行函数说明。之后,需要进行函数说明。类型名类型名 函数名函数名(
15、类型类型1 变量变量1,类型类型2 变量变量2,类型类型n 变量变量n);说明:说明:函数声明应与该函数定义的函数类型与名称、形参的个函数声明应与该函数定义的函数类型与名称、形参的个数、类型、次序相一致。数、类型、次序相一致。函数声明中的形参名可省略,其形式为函数声明中的形参名可省略,其形式为:类型名类型名 函数名(类型函数名(类型1,类型,类型2,类型,类型n););类型名类型名 函数名函数名();当函数定义在主调函数之前,即先定义当函数定义在主调函数之前,即先定义,后调用。则调用时后调用。则调用时函数声明可以省略。函数声明可以省略。21第21页,共45页,编辑于2022年,星期一例例6.4
16、 编写计算编写计算x的的n次乘方的程序。次乘方的程序。#include stdio.h”main()float x,y;int n;float power(float x,int n);scanf(%f,%d,&x,&n);y=power(x,n);printf(“%8.2f”,y);float power(float x,int n)int i;float t=1;for(i=1;iy)t=x;else t=y;return t;24第24页,共45页,编辑于2022年,星期一1)形参形参y之间值的传递如图之间值的传递如图4.7所示意所示意。a x b y2)关于形式参数和实际参数说明如下关
17、于形式参数和实际参数说明如下:形式参数在函数被调用时才被分配内存。当函数执行完毕返形式参数在函数被调用时才被分配内存。当函数执行完毕返回时回时,形式参数占用的内存空间便被释放。形式参数占用的内存空间便被释放。实参可以是变量、常量和表达式。实参可以是变量、常量和表达式。如如:y=power(x,4);y=power(x,i*2);但实参必须有确定的值。但实参必须有确定的值。3)形参和实参的类型必须相容形参和实参的类型必须相容。4)形参和实参之间的关系是形参和实参之间的关系是:单向的值的传递单向的值的传递10105525第25页,共45页,编辑于2022年,星期一6.3.3 函数的嵌套调用函数的嵌
18、套调用 调调用用一一个个函函数数的的过过程程中中又又调调用用了了另另一一个个函函数数,这这种种调调用用称称为为函函数数的的嵌嵌套套调用调用。函数函数1 函数函数2 函数函数3 .调用函数调用函数2 调用函数调用函数3 .26第26页,共45页,编辑于2022年,星期一【例【例6-9】求方程求方程ax2+bx+c=0(a 0)的根。的根。#include#include void main()float a,b,c,x1,x2;int dict(float,float,float);float root(float,float,float,int);printf(Input a,b,c:);sc
19、anf(%f,%f,%f,&a,&b,&c);if(dict(a,b,c)x1=root(a,b,c,1);/*调用函数调用函数root*/x2=root(a,b,c,0);printf(实根实根x1=%f,x2=%fn,x1,x2);else printf(无实根无实根!n);27第27页,共45页,编辑于2022年,星期一dict()和和root()int dict(float a,float b,float c)int f;if(b*b-4*a*c=0)f=1;else f=0return f;float root(float a,float b,float c,int flag)flo
20、at d,x;d=dict(a,b,c);/*调用函数调用函数 dict*/if(d)x=flag?(-b+sqrt(d)/(2*a):(-b-sqrt(d)/(2*a);return x;28第28页,共45页,编辑于2022年,星期一嵌套过程嵌套过程void main().x1=root(a,b,c,1)x2=root(a,b,c,0).root(a,b,c,flag).d=dict(a,b,c).return x;dict(a,b,c).return f;图图6-3 嵌套调用过程嵌套调用过程 调用调用 返回返回 返回返回 调用调用29第29页,共45页,编辑于2022年,星期一6.3.4
21、 递归调用递归调用(重点重点)函数调用函数本身,称为函数的递归调用。递归调用函数调用函数本身,称为函数的递归调用。递归调用形式如下:形式如下:1)直接递归直接递归 void a().a();.2)间接递归间接递归 void a().b();.void b().a();.30第30页,共45页,编辑于2022年,星期一用递归算法计算用递归算法计算n!讨论:讨论:采用递归的方法计算。采用递归的方法计算。n!的递归定义形式的的递归定义形式的:编程编程:if(初始条件初始条件)表达式表达式;else 递推表达式递推表达式;1 n=0;n=n*(n-1)!n031第31页,共45页,编辑于2022年,星
22、期一例程序:例程序:#include long fac(unsigned n)long f;if (n=0)f=1;/*递归结束条件递归结束条件*/else f=n*fac(n-1);return f;main()long y;int n;scanf(“%d”,&n);y=fac(n);printf(“%d!=%ldn”,n,y);32第32页,共45页,编辑于2022年,星期一分析分析:当程序输入当程序输入3时时y=fac(3)3*fac(2)2*fac(1)1*fac(0)133第33页,共45页,编辑于2022年,星期一例例6.11 汉诺塔游戏汉诺塔游戏汉诺塔汉诺塔(Tower of H
23、anoi)游戏。游戏。底座上有三根针,底座上有三根针,第一根针上放着从大到小第一根针上放着从大到小64个金片。游戏的目标个金片。游戏的目标是把所有金片从第一根针通过第二根针移到第三是把所有金片从第一根针通过第二根针移到第三根针上。移动过程中大的金片不能压在小的金片根针上。移动过程中大的金片不能压在小的金片上。上。把把n(n1)个金片从第一根针个金片从第一根针a上移到第三根针上移到第三根针c的的问题分解成如下步骤问题分解成如下步骤:(1)将将n-1个金片从个金片从a经过经过c 移动到移动到b。(2)将第将第n个金片移动到个金片移动到c。(3)再将再将n-1个盘子从个盘子从b经过经过a移动到移动到
24、c。34第34页,共45页,编辑于2022年,星期一void hanoi(int n,int a,int b,int c)if(n=0)return;/*0个金片不处理*/if(n=1)printf(%d-%dn,a,c);/*n=1时,直接将金片从a移动到c*/else hanoi(n-1,a,c,b);/*先将n-1个金片从a经过c 移动到b*/printf(%d-%dn,a,c);/*将第n个金片从a移动到c*/hanoi(n-1,b,a,c);/*再将n-1个金片从b经过a移动到c*/主函数如下:#include main()int n;printf(Input n:);scanf(%
25、d,&n);hanoi(n,1,2,3);/*n个金片从第一根针经过第二根针移动到第三根针上*/35第35页,共45页,编辑于2022年,星期一6.4 变量的作用域及存储特性变量的作用域及存储特性例例void f1()int t=2;a*=t;b/=t;main()int a,b;printf(“Enter a,b:”);scanf(“%d,%d”,&a,&b);f1();/*调用函数调用函数f1()*/printf(“a=%d,b=%d”,a,b);编译程序会提示出错编译程序会提示出错:Undefined symbol a 和和 Undefined symbol b。为什么。为什么?36第3
26、6页,共45页,编辑于2022年,星期一全局变量(外部变量)局部变量(内部变量)定义位置 函数体外函数体内作用域从定义处到文件结束从定义处到本函数结束举例所有在函数体外定义的变量(1)所有在函数体内定义的变量(2)形式参数注意与局部变量同名的处理不同函数中同名局部变量互不干扰6.4.1 变量的作用域变量的作用域1.变量按作用域:分为全局变量和局部变量2.区别:37第37页,共45页,编辑于2022年,星期一例例程序程序#include int a,b;/*a,b为全局变量为全局变量*/void f1()int t1,t2;t1=a*2;t2=b*3;b=100;printf(“t1=%d,t2
27、=%dn”,t1,t2);main()a=2;b=4;f1();printf(“a=%d,b=%d”,a,b);程序输出结果为程序输出结果为:t1=4,t2=12 a=2,b=100 38第38页,共45页,编辑于2022年,星期一将程序改为:将程序改为:#include int a=2,b=4;/*a,b为全局变量为全局变量*/void f1()int t1,t2;t1=a*2;t2=b*3;b=100;printf(“t1=%d,t2=%dn”,t1,t2);main()int b=4;f1();printf(“a=%d,b=%d”,a,b);程序输出结果为程序输出结果为:t1=4,t2=
28、12 a=2,b=4 结论:全局变量与局部结论:全局变量与局部变量同名时,局部变量变量同名时,局部变量的作用域屏蔽全局变量的作用域屏蔽全局变量39第39页,共45页,编辑于2022年,星期一6.4.2 变量的存储特性变量的存储特性1.变量按存在时间分变量按存在时间分:静态变量,动态变量静态存储变量:生存期为程序执行的整个过程,在该过静态存储变量:生存期为程序执行的整个过程,在该过程中占有固定的存储空间,也称永久存储。程中占有固定的存储空间,也称永久存储。动态存储变量:只生存在某一段时间内。动态存储变量:只生存在某一段时间内。例如:例如:函数的形参、函数体或分程序函数的形参、函数体或分程序中定义
29、的变量,只有中定义的变量,只有当程序进入该函数或分程序时才分配存储空间,函数当程序进入该函数或分程序时才分配存储空间,函数/分分程序执行完后,变量的存储空间又被释放。程序执行完后,变量的存储空间又被释放。2.变量属性:数据类型,存储特性变量属性:数据类型,存储特性完整的变量定义完整的变量定义:存储特性存储特性 数据类型数据类型 变量名变量名;40第40页,共45页,编辑于2022年,星期一3.变量的存储特性变量的存储特性1)auto型型有形式参数、函数内变量、分程序变量。有形式参数、函数内变量、分程序变量。进入程序自动分配内存,不长期占用内存。进入程序自动分配内存,不长期占用内存。2)stat
30、ic 型型 局部静态变量局部静态变量 全局静态变量全局静态变量长期占用内存长期占用内存自动型自动型 auto静态型静态型 static寄存器型寄存器型 register外部型外部型 extern41第41页,共45页,编辑于2022年,星期一例例f(int a)int b=0;static int c=3;b+;c+;printf(“%5d%5d%5d”,a,b,c);return(a+b+c);main()int a=2,k;for(k=0;k3;k+)printf(“%5dn”,f(a);(看看L4_11)42第42页,共45页,编辑于2022年,星期一3)register型型将使用频率高
31、的变量定义为将使用频率高的变量定义为register型型,可以提高运可以提高运行速度。行速度。寄存器变量只限于寄存器变量只限于整型整型、字符型字符型、指针型指针型的局部变量。寄存的局部变量。寄存器变量是动态变量,仅允许说明两个寄存器变量。器变量是动态变量,仅允许说明两个寄存器变量。例如例如:register int d;register char c;数据内存 运算器 运算器 结果 控制器 数据寄存器寄存器43第43页,共45页,编辑于2022年,星期一4)extern型型引用引用:extern 类型类型 变量名变量名;如果某个模块文件中要用到另一个模块文件中的全局变量,如果某个模块文件中要用
32、到另一个模块文件中的全局变量,要用要用extern说明。说明。例如例如:程序模块程序模块file1.c中定义了全局变量中定义了全局变量 int s;另一程序另一程序file2.c的函数的函数fun1()需要使用这个变量需要使用这个变量s。在。在file2.c的的fun1()对对s进行外部变量说明进行外部变量说明:fun1()extern int s;/*表明变量表明变量s是在其他文件定义的是在其他文件定义的*/.定义时分配内存定义时分配内存,其他文件引用时不再分配内存。其他文件引用时不再分配内存。44第44页,共45页,编辑于2022年,星期一练习练习:main()int i=1;static int a=10;register int b=5;printf(“i=%d,a=%d,b=%dn”,i,a,b);other();printf(“i=%d,a=%d,b=%dn”,i,a,b);other()int i;static int a;i=6;a=100;printf(“i=%d,a=%dn”,i,a);(看看L4_12)45第45页,共45页,编辑于2022年,星期一