C程序设计ch08 函数33866_第1页
C程序设计ch08 函数33866_第2页
C程序设计ch08 函数33866_第3页
C程序设计ch08 函数33866_第4页
C程序设计ch08 函数33866_第5页
已阅读5页,还剩54页未读 继续免费阅读

下载本文档

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

文档简介

1、第八章 函数 v8.1 概述 模块化程序设计 v若只有main函数,则程序太长、太复杂,导致跟踪、 调试、维护困难 v基本思想:将一个大的程序按功能分割成一些小模 块,各部分独立编码、调试、再集成。 v特点: 各模块相对独立、功能单一、结构清晰、接口简单 控制了程序设计的复杂性 易理解 易于维护和功能扩充 v开发方法: 自上向下,逐步分解,分而治之 C程序结构 lC是函数式语言 l必须有且只能有一个名为main的主函数 lC程序的执行总是从main函数开始,在main中结束 l函数不能嵌套定义,可以嵌套调用 C程序 源程序文件1源程序文件2源程序文件n 预编译命令函数1函数m 说明部分 执行部

2、分 函数分类 v从用户角度 标准函数(库函数):由系统提供 用户自定义函数 v从函数形式 无参函数 有参函数 使用库函数的一般方法: 1、函数功能 2、函数参数的数目和顺序,及各参数意义和类型 3、函数返回值意义和类型 4、需要使用的包含文件 v8.2 自定义函数 多函数程序 #include stdio.h main() void printstar(); printstar(); printf(This is a example of use C functionn); printstar(); void printstar() printf(*n); 函数控制流 函数与变量的对比 函数名

3、和变量名均以标识符体现 函数和变量均有对应的类型,如int 函数和变量一样,均要prototype 函数定义格式 函数体 合法标识符 函数返回值类型 缺省缺省int型型 无返回值void 函数类型 函数名(形参类型说明表) 说明部分 语句部分 return 语句 以上红色的为函数的6个元素 函数头部 函数类型 函数名(形参表) 形参类型说明 说明部分 语句部分 传统风格: 例 有参函数(传统风格) int max(x,y) int x,y; int z; z=xy?x:y; return(z); 函数类型 void printstar() printf(*); void swap(int x,

4、int y ) int temp; temp=x; x=y; y=temp; float squre(float x ) return (x*x); int get_number(void) int temp=1; return temp; int mathoperation(int a,int b,int *s) 返回多个数据 返回语句 v形式: return(表达式); 或 return 表达式; 或 return; v功能:使程序控制从被调用函数返回到调用函数中, 同时把返值带给调用函数 v说明: 函数中可有多个return语句 若无return语句,遇时,自动返回调用函数 若函数类型与

5、return语句中表达式值的类型不一致,按前 者为准,自动转换-函数调用转换 void型函数 例 无返回值函数 void swap(int x,int y ) int temp; temp=x; x=y; y=temp; return; printstar() printf(*); main() int a; a=printstar(); printf(%dn,a); 例 函数带回不确定值 输出:10 void printstar() printf(*); main() int a; a=printstar(); printf(%dn,a); 编译错误! 例 函数返回值类型转换 main()

6、float a,b; int c; scanf(%f,%f, c=max(a,b); printf(Max is %dn,c); max(float x, float y) float z; z=xy?x:y; return(z); 函数的调用 调用形式 函数名(实参表); 说明: 实参与形参个数相等,类型一致,按顺序一一对应 实参表求值顺序,因系统而定(Turbo C,VC+ 自右向左) #include stdio.h int f(int a, int b) int c; if(ab) c=1; else if(a=b) c=0; else c=-1; return(c); main()

7、int i=2,p; p=f(i,+i); printf(%dn,p); 例 参数求值顺序 #include stdio.h int f(int a, int b) int c; if(ab) c=1; else if(a=b) c=0; else c=-1; return(c); main() int i=2,p; p=f(i, i+); printf(%dn,p); 运行结果:0运行结果:1 调用方式 v函数语句: 例 printstar(); printf(“Hello,World!n”); v函数表达式: 例 m=max(a,b)*2; v函数参数: 例 printf(“%d”,max

8、(a,b); m=max(a,max(b,c); 说明 v对被调用函数要求: 必须是已存在的函数 库函数: #include 用户自定义函数: 函数类型说明 v函数说明 一般形式: 函数类型 函数名(形参类型 形参名,. ); 或 函数类型 函数名(); 作用:告诉编译系统函数类型、参数个数及类型,以便检验 函数说明位置:程序的数据说明部分(函数内或外) 下列情况下,可不作函数说明 v若函数返值是char或int型,系统自动按int型处理 v被调用函数定义出现在主调函数之前 有些系统(如Borland C+)要求函数说明指出函数返值类型和形 参类型,并且对void 和 int 型函数也要进行函

9、数说明 例 函数说明举例 main() float a,b; int c; scanf(%f,%f, c=max(a,b); printf(Max is %dn,c); max(float x, float y) float z; z=xy?x:y; return(z); int型函数可不作函数说明 (VC+不行) /*/ float add(float x, float y) float z; z=x+y; return(z); main() float a,b,c; scanf(%f,%f, c=add(a,b); printf(sum is %f,c); 被调函数出现在主调函数 之前,不

10、必函数说明 main() float add(float,float); /*function declaration*/ float a,b,c; scanf(%f,%f, c=add(a,b); printf(sum is %f,c); float add(float x, float y) float z; z=x+y; return(z); float add(); 函数参数及其传递方式 v形式参数:定义函数时函数名后面括号中的变量名 v实际参数:调用函数时函数名后面括号中的表达式 c=max(a,b); (main 函数) (max 函数) max(int x, int y) int

11、 z; z=xy?x:y; return(z); 例 比较两个数并输出大者 main() int a,b,c; int max(int, int); scanf(%d,%d, c=max(a,b); printf(Max is %d,c); max(int x, int y) int z; z=xy?x:y; return(z); 形参 实参 v说明: 实参必须有确定的值 形参必须指定类型 形参与实参类型一致,个数相同 若形参与实参类型不一致,自动按形参类型转换函数 调用转换 形参在函数被调用前不占内存;函数调用时为形参分配内存; 调用结束,内存释放 例 计算x的立方 #include flo

12、at cube(float x) return(x*x*x); main() float a, product; printf(Please input value of a:); scanf(%f, product=cube(a); printf(”Cube of %.4f is %.4fn,a,product); x a product 1.2 1.2 1.728 v参数传递方式 方式:函数调用时,为形参分配单元,并将实参的值复制到形参 中;调用结束,形参单元被释放,实参单元仍保留并维持原值 特点: v形参与实参占用不同的内存单元 v单向传递值传递 711x:y:调用前: 调用结束:711

13、x:y: 例 交换两个数 /*swap a pair number*/ #include main() int x=7,y=11; int swap(int, int); printf(x=%d,ty=%dn,x,y); printf(swapped:n); swap(x,y); printf(x=%d,ty=%dn,x,y); swap(int a,int b) int temp; temp=a; a=b; b=temp; 调用: 711a:b: 711x:y: swap: 711x:y: 117a:b: temp v地址传递 方式:函数调用时,将数据的存储地址作为参数传递给形参 特点: v

14、形参与实参占用同样的存储单元 v“双向”传递 v实参和形参必须是地址常量或变量 /*call by address*/ swap(int *p1,int *p2) int p; p=*p1; *p1=*p2; *p2=p; main() int a,b; int swap(int *, int *); scanf(%d,%d, printf(“a=%d,b=%dn”,a,b); printf(“swapped:n”); swap( printf(”a=%d,b=%dn,a,b); 例 交换两个数 a 59 b 调前: a 5 9 b 调swap: p1 long factorial(int )

15、; main() int n1,n2; long a; scanf(%d,%d, a=sum(n1,n2); printf(a=%1d,a); long sum(int a,int b) long c1,c2; c1=factorial(a); c2=factorial(b); return(c1+c2); long factorial(int n) long rtn=1; int i; for(i=1;ibi, ai=bi, aik,认为数组ab 若nk,认为数组ab 若n=k,认为数组a=b #include main() int a10,b10,i,n=0,m=0,k=0; int la

16、rge(int ,int); printf(Enter array a:n); for(i=0;i10;i+) scanf(%d, printf(Enter array b:n); for(i=0;i10;i+) scanf(%d, for(i=0;iy) flag=1; else if(xy) flag=-1; else flag=0; return(flag); v数组名作函数参数 地址传递 形参数组大小(多维数组第一维)可不指定。常用方法 是用另一参数传递数组长度 形参数组名是地址变量 例 求学生的平均成绩 #include float average(int , int); void

17、main() int score10, i; float av; printf(Input 10 scores:n); for( i=0; i10; i+ ) scanf(%d, 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; 实参用数组名 形参用数组定义, int stu . . 2 1 0 9 score 5

18、6 23 12 . . 88 stu 例 数组元素与 数组名 作函数参数比较 1 2 a 调用前 a0 a1 1 2 a 调用 a0 a1 1 2 x y 2 1 x y 交换 1 2 a 返回 #include void swap2(int x,int y) int z; z=x; x=y; y=z; main() int a2=1,2; void swap2(int, int); swap2(a0,a1); printf(a0=%dna1=%dn,a0,a1); 值传递 1 2 a 调用前 1 2 a x 调用 2 1 a x 交换 2 1 a 返回 #include void swap2

19、(int x) int z; z=x0; x0=x1; x1=z; main() int a2=1,2; void swap2(int ); swap2(a); printf(a0=%dna1=%dn,a0,a1); 地址传递 例 数组元素与 数组名 作函数参数比较 例 数组排序-简单选择排序 #include stdio.h void sort(int array,int n) int i,j,k,t; for(i=0;in-1;i+) k=i; for(j=i+1;jn;j+) if(arrayjarrayk) k=j; if(k!=i) t=arrayi; arrayi=arrayk;

20、arrayk=t; main() int a10,i; void sort(int ,int); for(i=0;i10;i+) scanf(%d, sort(a,10); for(i=0;i10;i+) printf(%d ,ai); printf(n); 0 1 2 3 4 5 6 7 8 9 a 49 68 57 32 9 99 27 13 76 88 array k j j jk jk j j j j j 9 49 i=0 例 数组排序-简单选择排序 #include stdio.h void sort(int array,int n) int i,j,k,t; for(i=0;in-

21、1;i+) k=i; for(j=i+1;jn;j+) if(arrayjarrayk) k=j; if(k!=i) t=arrayi; arrayi=arrayk; arrayk=t; main() int a10,i; void sort(int ,int); for(i=0;i10;i+) scanf(%d, sort(a,10); for(i=0;i10;i+) printf(%d ,ai); printf(n); k j jk j k j j j j j 0 1 2 3 4 5 6 7 8 9 a 49 68 57 32 9 99 27 13 76 88 array 9 49 k k

22、 13 68 i=1 0 1 2 3 4 5 6 7 8 9 a 9 13 27 32 49 57 68 76 88 99 array i=8 例 数组排序-简单选择排序 #include stdio.h void sort(int array,int n) int i,j,k,t; for(i=0;in-1;i+) k=i; for(j=i+1;jn;j+) if(arrayjarrayk) k=j; if(k!=i) t=arrayi; arrayi=arrayk; arrayk=t; main() int a10,i; void sort(int ,int); for(i=0;i10;i

23、+) scanf(%d, sort(a,10); for(i=0;i10;i+) printf(%d ,ai); printf(n); 例 求二维数组中最大元素值 1 3 5 7 2 4 6 8 15 17 34 12 i j max=1 1 3 5 7 2 4 6 8 15 17 34 12 i j max=3 1 3 5 7 2 4 6 8 15 17 34 12 i j max=5 j 1 3 5 7 2 4 6 8 15 17 34 12 i max=7 j 1 3 5 7 2 4 6 8 15 17 34 12 i max=7 j 1 3 5 7 2 4 6 8 15 17 34 1

24、2 i max=34 #include stdio.h int 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; int max_value(int 4); printf(max value is %dn,max_value(a); 按ANSI C,多维形参数组第一维维数 可省略,第二维必须相同 int array 4 在在VC+中,要求用二级指

25、针中,要求用二级指针 例 求二维数组中各行元素之和 get_sum_row(int x3, int result ,int row, int col) int i,j; for(i=0;irow;i+) resulti=0; for(j=0;jcol;j+) resulti+=xij; main() int a23=3,6,9,1,4,7; int sum_row2,row=2,col=3,i; get_sum_row(a,sum_row,row,col); for(i=0;irow;i+) printf(The sum of row%d=%dn,i+1,sum_rowi); 3 14 6 7

26、 9 asum_row x result 18 12 v8.3 函数的嵌套与递归调用 嵌套调用 p271 C规定:函数定义不可嵌套,但可以嵌套调用函数 main( ) 调用函数a 结束 a函数b函数 调用函数b 例 求三个数中最大数和最小数的差值 #include int dif(int x,int y,int z); int max(int x,int y,int z); int min(int x,int y,int z); void main() int a,b,c,d; scanf(%d%d%d, d=dif(a,b,c); printf(Max-Min=%dn,d); int dif

27、(int x,int y,int z) return max(x,y,z)-min(x,y,z); int max(int x,int y,int z) int r; r=xy?x:y; return(rz?r:z); int min(int x,int y,int z) int r; r=xy?x:y; return(rz?r:z); main( ) 调用函数dif 输出 结束 dif函数max函数 调用函数max 调用函数min min函数 例 用弦截法求方程根080165 23 xxx x y f(x) 0 x1 x2 x f(x1) f(x2) )()( )()( 12 1221 xf

28、xf xfxxfx x 求f(x1)与f(x2)连线与x轴的交点x 输入x1,x2,求f(x1),f(x2) 直到f(x1)与f(x2)异号 y=f(x),y1=f(x1) y与y1同号 真假 x1=x y1=y x2=x 直到 |y| root=x 输出 root root函数 运行情况: Input x1,x2: 2,6 A root of equation is 5.0000 main( ) 调用函数root 输出根 x 结束 root函数xpoint函数 调用函数xpoint调用函数f f函数 递归调用 p272 v定义:函数直接或间接的调用自身叫函数的递归调用 f( ) 调f 调f2

29、调f1 f1( )f2( ) v说明 C编译系统对递归函数的自调用次数没有限制 每调用函数一次,在内存堆栈区分配空间,用于存放函数变 量、返回值等信息,所以递归次数过多,可能引起堆栈溢出 int f(int x) int y,z; z=f(y); . return(2*z); int f1(int x) int y,z; z=f2(y); . return(2*z); int f2(int t) int a,c; c=f1(a); . return(3+c); 例 求n的阶乘 ) 1()!1( ) 1 , 0(1 ! nnn n n /*factorial*/ #include int fac

30、(int n) int f; if(n0) printf(n%cn,getone,putone); void hanoi(int n,char one,char two,char three) if(n=1) move(one,three); else hanoi(n-1,one,three,two); move(one,three); hanoi(n-1,two,one,three); main() int m; printf(Input the number of disks:); scanf(%d, printf(The steps to moving %3d disks:n,m); h

31、anoi(m,A,B,C); ABC v6.4 变量的存储属性 p279 概述 v变量是对程序中数据的存储空间的抽象内存 . main() int a; a=10; printf(“%d”,a); 编译或函数调用时为其分配内存单元 10 2000 2001 程序中使用变量名对内存操作 v变量的属性 数据类型:变量所持有的数据的性质(操作属性) 存储属性 v存储器类型:寄存器、静态存储区、动态存储区 v生存期:变量在某一时刻存在-静态变量与动态变量 v作用域:变量在某区域内有效-局部变量与全局变量 v变量的存储类型 auto -自动型 register-寄存器型 static -静态型 exte

32、rn -外部型 v变量定义格式: 存储类型 数据类型 变量表; 如: int sum; auto int a,b,c; register int i; static float x,y; 局部变量与全局变量 v局部变量-内部变量 规则:在函数内定义,只在本函数内有效(main中定义的变量 也只在main中有效) 说明: v不同函数中同名变量,占不同内存单元 v形参属于局部变量 v可定义在复合语句中有效的变量 v局部变量可用存储类型:auto register static (默认为auto) float f1(int a) int b,c; . char f2(int x,int y) int

33、 i,j; main() int m,n; . a,b,c有效 x,y,i,j有效 m,n有效 运行结果:5 4 3 2 1 例 复合语句中变量 #define N 5 main() int i; int aN=1,2,3,4,5; for(i=0;iN/2;i+) int temp; temp=ai; ai=aN-i-1; aN-i-1=temp; for(i=0;iN;i+) printf(%d ,ai); 例 不同函数中同名变量 main() int a,b; int sub(); a=3; b=4; printf(main:a=%d,b=%dn,a,b); sub(); printf(

34、main:a=%d,b=%dn,a,b); sub() int a,b; a=6; b=7; printf(sub:a=%d,b=%dn,a,b); 运行结果: main:a=3,b=4 sub:a=6,b=7 main:a=3,b=4 v全局变量-外部变量 规则:在函数外定义,可为本文件所有函数共用 有效范围:从定义变量的位置开始到本源文件结束,及有 extern说明的其它源文件 应尽量少使用全局变量,因为: Y全局变量在程序全部执行过程中占用存储单元 Y降低了函数的通用性、可靠性,可移植性 Y降低程序清晰性,容易出错 定义 说明 u次数: 只能1次 可说明多次 u位置: 所有函数之外 函数

35、内或函数外 u分配内存: 分配内存,可初始化 不分配内存,不可初始化 外部变量说明: extern 数据类型 变量表; 外部变量定义与外部变量说明不同 若外部变量与局部变量同名,则外部变量被屏蔽 外部变量可用存储类型:缺省 或 static #include stdio.h float max,min; float average(float array, int n) int i; float sum=array0; max=min=array0; for(i=1;imax) max=arrayi; else if(arrayiy?x:y; return(z); #include stdio

36、.h main() extern int a,b; int max(int,int); printf(max=%dn,max(a,b); int a=13,b=-8; 运行结果:max=13 extern int a,b; int max() int z; z=ab?a:b; return(z); #include stdio.h main() printf(max=%d,max(); int a=13,b=-8; #include stdio.h int a=3,b=5; max(int a, int b) int c; c=ab?a:b; return(c); main() int a=8

37、; printf(max=%dn,max(a,b); 例 外部变量与局部变量 运行结果:max=8 #include stdio.h int i; main() void prt(); for(i=0;i5;i+) prt(); void prt() for(i=0;i5;i+) printf(“%c”,*); printf(“n”); 例 外部变量副作用 运行结果:* 动态变量与静态变量 v存储方式 静态存储:程序运行期间分配固定存储空间 动态存储:程序运行期间根据需要动态分配存储空间 v内存用户区 程序区 静态存储区 动态存储区 全局变量、局部静态变量 形参变量 局部动态变量(auto r

38、egister) 函数调用现场保护和返回地址等 v生存期 静态变量:从程序开始执行到程序结束 动态变量:从包含该变量定义的函数开始执行至函数执行结束 变量存储类型 静态动态 存储方式 程序整个运行期间函数调用开始至结束生存期 编译时赋初值,只赋一次 每次函数调用时 赋初值 自动赋初值0或空字符 不确定未赋初值 静态存储区 动态区存储区寄存器 局部变量外部变量 作用域定义变量的函数或复合语句内本文件其它文件 u局部变量默认为auto型 u局部static变量具有全局寿命和局部可见性 u局部static变量具有可继承性 uextern不是变量定义,可扩展外部变量作用域 register局部stat

39、icauto外部static外部存储类别 例 文件file1.c int a; main( ) . . f2; . f1; . f1( ) auto int b; f2; . f2( ) static int c; C作用域 b作用域 a作用域 mainf2f1mainf1f2main a生存期: b生存期: c生存期: 例 auto 变量的作用域 main() int x=1; void prt(void); int x=3; prt(); printf(“2nd x=%dn”,x); printf(“1st x=%dn”,x); void prt(void) int x=5; printf

40、(“3th x=%dn”,x); 运行结果: 3th x=5 2nd x=3 1st x=1 x=1作用域 x=1作用域 x=3作用域 x=5作用域 main() void increment(void); increment(); increment(); increment(); void increment(void) int x=0; x+; printf(“%dn”,x); 例 局部静态变量值具有可继承性 运行结果:1 1 1 main() void increment(void); increment(); increment(); increment(); void increment(void) static int x=0; x+; printf(“%dn”,x); 运行结果:1 2 3 例 变量的寿命与可见性 #include int i=1; main() static int a; register int b=-10; int c=0; printf(-MAIN-n); printf(i:%d a:%d b:%d c:%dn,i,a,b,c); c=c+8; other(); printf(-MAIN-n); printf(i:%d a:%d b

温馨提示

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

评论

0/150

提交评论