《C语言程序设计课件第5章.ppt》由会员分享,可在线阅读,更多相关《C语言程序设计课件第5章.ppt(56页珍藏版)》请在三一办公上搜索。
1、返回主目录,C语言程序设计,(第5章 函数),本章主要介绍C语言程序函数的基本知识和基本应用。,1.函数与模块化程序设计2.函数的概述3.函数的嵌套调用和递归调用4.变量的作用域和生存期5.编译预处理,一、函数与模块化程序设计,1 模块化程序设计方法:,程序员在设计一个比较复杂的应用程序时,一般采用的方法是将整个程序分解成若干个功能较为单一的、相对独立的程序模块分别实现,然后再将所有的程序模块象搭积木一样装配起来,形成一个完整的程序,这种在程序设计中分而治之的策略称为模块化程序设计方法。,优点:主要体现在程序编制方便,易于修改和调试,可由多人共同参与完成,缩短程序的设计时间,并能实现程序模块的
2、标准化、规范化。,在C语言中,函数是程序的基本组成单位,程序设计人员可以很方便地用函数作为程序模块来实现程序的模块化,一个函数就是一个模块,独立完成一项功能。,采用普通的程序设计方法#include stdio.hmain()int a,b,max;printf(输入两个整数:);scanf(%d,%d,例5-1 求两个整数的最大值。,采用结构化程序设计方法#include stdio.hint fmax(int x,int y)int z;if(xy)z=x;else z=y;return z;main()int a,b,max;printf(输入两个整数:);scanf(%d%d,2 程序
3、模块设计一般原则,模块相对独立 一个模块往往要求完成一个单独的功能,并与其他模块尽量保持一定的独立性,当修改这个模块时,不会引起整个程序的混乱。同时,在模块内部,尽量将数据局部化,便于保证数据在不同模块之间的正常传递。,模块之间关系尽量简单 模块之间关系要尽量简单,且不互相干扰,做到模块之间只有在相互调用时才发生数据传递。除此之外,相互之间不直接发生联系。,模块规模大小要适中 程序模块不宜过大,也不要过于复杂,大小要适中。模块过大,容易造成程序功能紊乱,可读性差,不好理解。模块过小,容易造成程序结构复杂,数据传递容易出错。,模块尽量保持通用性 模块应尽量规范,保持一定的通用性,这样便于对模块进
4、行修改和维护,快速实现对程序的扩充。,一、函数的分类与定义,函数可以分为无参函数 和有参函数。,函数可以分为标准函数 和自定义函数。,有一个或多个形参,无形参,由系统提供,可直接使用,由用户自己定义使用,二、函数的定义,1无参函数定义的一般形式:函数类型说明符 函数名()函数体,例如:printstar()printf(*n);,2有参函数定义的一般形式:函数类型说明符 函数名(形参表列)函数体,例如:int fmax(int x,int y)int z;if(xy)z=x;else z=y;return z;,说明:C语言规定:函数不允许嵌套定义。函数之间是平行的、相对独立的。一个函数的定义
5、可 以放在程序中的任何一个位置,但在一个函数的函数体内,不能再定义另外一个函数。函数的类型是指函数返回值的类型,可以是整型、实型、字符型、指针类型等等数据类型,如果函数没有返回值,则函数类型为void。函数名必须符合标识符的命名规则,函数名不能与变量名重名,也不能与关键字同名。函数体包含两部分:语句声明部分和执行语句部分。空函数既无参数,又无函数体,其一般形式为:函数类型说明符 函数名(),函数的调用在定义一个函数后,要调用该函数才能执行该函数的功能,否则,该函数在程序中只是一段静态的代码,不起任何作用。,1.函数调用的一般形式 函数名(实参表列),注意:实参表列的形式为:实际参数1,实际参数
6、2,实参可以是常量或变量,也可以是各种表达式。实际参数的个数和类型应该与函数定义时的形式参数的个数和类型一致,并一一对应。例如:c=max(a,b);调用无参函数时,不能有实参表列。,三、函数的调用,2.函数调用的具体方式 函数语句把函数调用作为一个语句。其一般形式为:函数名(实际参数表);该方式常用于调用一个可以忽略返回值或没有返回值的函数。例如程序中的scanf()函数和printf()函数的调用。,函数表达式 函数调用出现在表达式中,这种表达式称为函数表达式。其一般形式为:变量名=函数表达式;这种方式用于调用带返回值的函数,函数的返回值将参加表达式的运算。例如程序中的max=maxnum
7、(i,j,k);,函数实参 函数作为另一个函数调用的实参出现,这种方式是把该函数的返回值作为实参进行传送,因此,要求该函数必须有返回值。例如:max=fmax(a,fmax(b,c);,四、函数的说明,C语言规定:函数调用之前必须对该函数进行定义或说明。,函数的定义可以写在程序中的任何位置,但如果被调函数定义在主调函数之后,则在主调函数中必须要对被调函数进行说明,这样在函数调用的时候才能够找到函数原型。,函数原型的一般形式是:函数类型说明符 函数名(参数类型1,参数类型2,)例如:int maxnum(int,int,int);函数类型说明符 函数名(参数类型1 参数名1,参数类型 2 参数名
8、2,)例如:int maxnum(int x,int y,int z);,五、函数的参数,在函数调用的过程中,两个函数之间的数据传递是靠参数和返回值来传递的,主调函数利用参数将数据传给被调函数,被调函数的运行结果通过返回值传回给主调函数。,#include stdio.hmain()int x,y,z;scanf(%d,%d,int max(int a,int b)int c;if(ab)c=a;else c=b;return c;,在调用函数时,函数名后面括号中的数据称为实际参数(简称实参),实参与形参的区别,在定义函数时函数名后面的括号中的变量称为形式参数(简称形参)。,在被调函数中出现,
9、在主调函数中出现。,变量,变量、常量、表达式,具体的数值,存储数值的空间,单向值传递,说明:形参在被调函数中出现,实参在主调函数中出现。形参是在函数被调用时临时分配存储单元的,一旦调用结束,形参所占的存储单元立即被释放。实参可以是常量、变量或表达式,但必须是确定的值;形参必须是变量。因为实参是函数在调用刚开始时传递的具体数据,所以必须是确定的值;形参是用来接受数据的,只有在内存中占有存储空间才能存放数据,因此,形参必须是变量。在定义函数时,必须指定形参类型。形参实际上就是定义的变量,如果不指定变量的类型,将无法分配内存单元。,实参传递给形参的实际值必须与函数定义中的形参类型一致。不一致时要在主
10、调函数中对被调函数作说明。调用语句中的实参个数应与被调用函数的形参个数相等,其类型、顺序必须一一对应。实参和形参在内存里占用不同的存储空间,即使同名也不会相互影响。C语言规定:实参对形参的传递是值传递,即单向传递,只能由实参传给形参,而不能由形参传给实参。,六、函数的返回值,任何函数都是有值的,如果值是恒定的,可以返回,则有返回值;如果值不恒定,无法返回,则没有返回值。函数的返回值是用return语句来实现的。,return语句的格式有两种:return r;或return(r);其中r为返回值,且从该函数返回。r可以是常量、变量、表达式,也可以是函数调用。,注意:如果函数没有返回值,可以事先
11、说明该函数的类型 是void型(即空类型),说明:函数类型应与返回值的类型保持一致,如果类型不一致,则以函数类型为准。若函数中无return语句,并不是不带回返回值,只是不带回有用的值,所以若不需要有返回值,则函数类型说明符为void,以明确表示不带回值。一个函数一次只能返回一个返回值。,例5-2,一、函数的嵌套调用,C语言中,函数的定义是相对独立的、平行的,没有隶属关系,函数不能嵌套定义,但可以嵌套调用。即在调用一个函数的过程中,又可以调用另外的函数。,例 5-3 求S=1K+2K+3K+nk的值,分析:本题可以用三个不同的函数来分别描述。f1函数计算nk f2函数求S=1K+2K+3K+n
12、k main()函数输入、输出初始条件和最后结果。,二、函数的递归调用,在函数的调用过程中,允许直接或间接地调用函数自身,这种调用称为递归调用。,函数直接自己调用自己,为直接递归。,函数间接自己调用自己,为间接递归。,例5-3 求整数n!,#include stdio.hlong fact(int n);main()int i;long fac;printf(i=);scanf(%d,例5-4 已知某数列为K(n)的定义如下,求该数列的第六项K(6)。1 n=1K(n)=K(n-1)*2 n为偶数 K(n-1)*3 n为奇数,main()int i;float j;printf(input t
13、he number:);scanf(%d,#include stdio.hfloat k(int n)float m;if(n=1)m=1.0;else if(n%2=0)m=k(n-1)*2;else m=k(n-1)*3;return m;,例5-5 有5个人坐在一起,问第5个人几岁,他说比第4个人大3岁,第4个人说比第3个人大3岁,第3个人说比第2个人大3岁,第2个人说比第1个人大3岁,第1个人说自己18岁。编程求第5个人几岁?,#include Stdio.hint age(int x)int y;if(x=1)y=18;else y=age(x-1)+3;return y;main(
14、)printf(%d,age(5);getch();,#include Stdio.hfloat p(int n,int x)float pp;if(n=0)pp=1;else if(n=1)pp=x;else pp=(2*x-1)*p(n-1,x)*x-(n-1)*p(n-1,x)/n;return pp;main()float f;int n,x;printf(input the n,x:);scanf(%d,%d,一、变量的作用域和生存期,变量的作用域,变量的生存期,变量在程序中影响范围,变量在程序中起作用的时间,长期性,暂时性,全局变量,局部变量,局部变量,变量只能在当前函数或 当前复
15、合语句中有效。形式参数是局部变量。可以在复合语句中定义 局部变量。局部变量可以同名,互 不干扰。,全局变量,在整个程序中有效有效。全局变量可以和局部变量同名,互不干扰,全局变量之间不能同名。在同一个程序中,如果全局变量与局部变量同名,在局部变量的作用范围内,全局变量暂时不起作用,被屏蔽。通常情况,建议尽量少使 用全局变量,#include stdio.hint a=3,b=5;max(int a,int b)int c;c=ab?a:b;return c;main()int a=8;printf(%d,max(a,b);,二、变量的存贮类别,数据的动态存贮方式和静态存贮方式,存放程序指令,存放
16、暂时性数据(局部变量),存放数据长期性(全局变量),在C语言中变量的存贮类别有四种,自动型变量auto出现范围:出现在函数内部。判别方法:定义在函数内部并且缺省存储类别说明 直接auto显式说明作用域:定义该变量的函数或复合语句。生存期:定义该变量的函数或复合语句被执行期间,是局部。存贮区域:动态存贮区注意:自动变量一般省略auto,函数中的大多数变量都是自动变量。形参和复合语句里定义的变量也是自动变量。自动变量一旦被说明,则只能在说明该变量的函数或复合语句 中出现才有效,离开了上述区域则无效。生存期只限于相应函数被调用时,若变量是定义在复合语句 内,则其生存期为该复合语句被执行的期间。,例
17、5-7#include stdio.hmain()int a=1;f(a);f(a);int f(int a)int b;b=a+;printf(%2d,b);,/*a是局部变量,作用域为main函数内*/,/*a、b都是局部变量,作用域为f函数内,生存期在调用f函数时*/,例5-8#include stdio.hmain()int a=3;int a=2;printf(%2d,a+);int a=0;printf(%2d,a+);printf(%2d,a+);,外部型变量extern出现范围:可以出现在程序中的任何说明部分。判别方法:在函数外部定义,并且缺省extern的变量 用extern
18、显式说明的变量。作用域:从定义该变量的位置起直到本文件结束。生存期:整个程序的执行期。存贮区域:静态存贮区域注意:外部型变量长驻数据的静态存贮区。外部变量在编译时创建,它只能初始化一次,如 果在定义时没有初始化,系统默认值为0。外部变量能被同名的内部变量所屏蔽。,例5-9#include stdio.hint a,b;main()a=1;fun();fun();int fun()b=a+;printf(%5d,b);,例5-10#include stdio.hint a,b;main()a=1;fun(a);fun(a);int fun(int a)b=a+;printf(%5d,b);,静态
19、变量static静态变量分成静态局部变量和静态全局变量。,注意:静态局部变量的作用域在定义该变量的函数内,生存期 是整个程序的执行期。要改变该变量的值,必须在函数 内部完成,相对于外部变量而言,静态变量更安全,起到 了信息屏蔽的作用,因此,静态变量的应用比较广泛。,静态变量初始化语句只在第一次调用该函数时才执行,以后每次调用该函数就不再执行初始化语句。,静态变量只能初始化一次,如果没有初始化,则编译自 动赋初值0,例5-11#include stdio.hint fun(int a)int b=0;static int c=3;b+;c+;return(a+b+c);main()int a=2
20、,i;for(i=0;i3;i+)printf(%5d,fun(a);getch();,例5-12#include stdio.hint fun(int n)static int f=1;f=f*n;return f;main()int i;for(i=1;i=5;i+)printf(%d!=%dn,i,fun(i);getch();,寄存器变量register出现范围:出现在函数内部。判别方法:用register声明的、在函数内部的变量。作用域:在定义该变量的函数内。生存期:定义该变量的函数被调用的期间。存贮区域:寄存器变量不是存放在内存里,而是存放在CPU 的寄存器里,它的特点是存取速度快
21、,如果变量存取很频繁,比如循环变量,则可以定义成寄存器变量以提高程序的执行速度。注意:只有自动变量和形参可以作为寄存器变量。静态变量和外部变量不能定义为寄存器变量。,例5-13#include stdio.hint sum(int n)register int i;int s=0;for(i=1;i=n;i+)s+=i;return s;main()int n;scanf(%d,在编写程序时可以利用C语言的预处理功能对程序进行编译处理。C语言提供的预处理功能主要有:宏定义、文件包含、条件编译。,一、编译预处理,不带参数的宏定义的一般形式为:#define 标识符 字符串例如:#define P
22、I 3.14159267,1.不带参数的宏定义,注意:宏名可以用大写字母书写,也可以用小写字母书写。宏定义是用宏名代替一个字符串,这里只是做一个简单的置 换,不进行数据检查 宏定义不是语句,后面不能加分号(;)宏定义通常出现在函数的外部。宏定义与变量定义不同,宏定义只作字符替换,不分配存储 空间。,例-14#include stdio.h#define x 5#define y x+1#define z y*x/2.0main()int a;a=y;printf(%f,%d,z,-a);,带参数宏定义的一般形式为:#define 宏名(形参表列)字符串,2.带参数的宏定义,带参数宏调用的一般形
23、式为:宏名(实数表列),例如:#define L(x)x*x+3*x+2 Y=L(5)前一个为宏定义,用字符串x*x+3*x+2代替L(x),即L(x)的值为x*x+3*x+2的值。后一个为宏调用,y=L(5)的值为*5+3*5+2。,例5-15include stdio.h#define MAX(x,y)xy?x:ymain()int x,y,z;scanf(%d,%d,注意:宏名和形参表列之间不能有空格出现。形参应为变量,实参可以是常量、也可以是具体表达式。为了避免出错,通常用括号将宏定义中形参括起来。例如:#define fun(x)y*y 如果y=fun(x+1);经过宏替换后,y的值为x+1*x+1。#define fun(x)(y)*(y)如果y=fun(x+1);经过宏替换后,y的值为(x+1)*(x+1)。可以看出:上面两种情况下的y 值明显不同。,3.文件包含处理,文件包含的一般形式如下:#include 文件名,例如:#include file1.h“就是将源文件file1.h的全部内容包含到当前文件中来,形成 一个新的源文件,这样便于编译链接。,如果在程序中使用了标准库函数,应在程序的开头用#include将其头文件包含进来,否则,编译链接时将会出现错误。,C语言程序允许文件包含嵌套。,普通函数设计方法,文件包含处理设计方法,