本章要点函数的概念函数的定义与调用函数的递归调用变量的作用域函数的作用域主要内容8.1概述8.2函数定义的一般形式8.3函数参数和函数的值8.4函数的调用8.5函数的嵌套调用8.6函数的递归调用8.7数组作为函数参数8.8局部变量和全局变量8.9变量的存储类别8.10内部函数和外部函数8.1概述一个C程序可由一个主函数和若干个其他函数构成。一个较大的程序可分为若干个程序模块,每一个模块用来实现一个特定的功能。在高级语言中用子程序实现模块的功能。子程序由函数来完成。函数间的调用关系:由主函数调用其他函数,其他函数也可以互相调用。同一个函数可以被一个或多个函数调用任意多次。例8.1先举一个函数调用的简单例子#includestdio.hvoidprintstar();/*对printstar函数声明*/voidprint_message();/*对print_message函数声明*/voidmain(){printstar();/*调用printstar函数*/print_message();/*调用print_message函数*/printstar();/*调用printstar函数*/}voidprintstar()/*定义printstar函数*/{printf(****************\n);}voidprint_message()/*定义print_message函数*/{printf(Howdoyoudo!\n);}运行情况如下:****************Howdoyoudo!****************说明:(1)一个C程序由一个或多个程序模块组成,每一个程序模块作为一个源程序文件。对于较大的程序,通常将程序内容分别放在若干个源文件中,再由若干源程序文件组成一个C程序。这样便于分别编写、分别编译,提高调试效率。一个源程序文件可以为多个C程序公用。(2)一个源程序文件由一个或多个函数以及其他有关内容(如命令行、数据定义等)组成。一个源程序文件是一个编译单位,在程序编译时是以源程序文件为单位进行编译的,而不是以函数为单位进行编译的。(3)C程序的执行是从main函数开始的,如果在main函数中调用其他函数,在调用后流程返回到main函数,在main函数中结束整个程序的运行。(4)所有函数都是平行的,即在定义函数时是分别进行的,是互相独立的。一个函数并不从属于另一函数,即函数不能嵌套定义。函数间可以互相调用,但不能调用main函数。main函数是系统调用的。(5)从用户使用的角度看,函数有两种:①标准函数,即库函数。这是由系统提供的,用户不必自己定义这些函数,可以直接使用它们。不同的C系统提供的库函数的数量和功能会有一些不同,但许多基本的函数是共同的。②用户自己定义的函数。用以解决用户的专门需要。(6)从函数的形式看,函数分两类:①无参函数。无参函数一般用来执行指定的一组操作。在调用无参函数时,主调函数不向被调用函数传递数据。②有参函数。主调函数在调用被调用函数时,通过参数向被调用函数传递数据。8.2函数定义的一般形式8.2.1无参函数的定义一般形式定义无参函数的一般形式为:类型标识符函数名(){声明部分语句部分}voidmain(){……}8.2.2有参函数定义的一般形式定义有参函数的一般形式为:类型标识符函数名(形式参数表列){声明部分语句部分}例如:intmax(intx,inty){intz;/*函数体中的声明部分*/z=x>y?x∶y;return(z);}8.2.3空函数定义空函数的一般形式为:类型标识符函数名(){}例如:dummy(){}主调函数调用空函数时,只表明这里要调用一个函数,但函数本身什么工作也不做,以后扩充函数功能时补充上。8.3函数参数和函数的值8.3.1形式参数和实际参数形式参数:定义时函数名后面括号中的变量名称为“形式参数”(简称“形参”)。实际参数:主调函数中调用一个函数时,函数名后面括号中的参数(可以是一个表达式)称为“实际参数”(简称“实参”)。函数返回值:return后面的括号中的值作为函数带回的值(称函数返回值)。例8.2调用函数时的数据传递#includestdio.hintmax(intx,inty);/*对max函数的声明*/voidmain(){inta,b,c;scanf("%d,%d",&a,&b);c=max(a,b);printf("Maxis%d",c);}intmax(intx,inty)/*定义有参函数max*/{intz;z=x>y?x∶y;return(z);}运行情况如下:7,8↙Maxis8通过函数调用,可使两个函数中的数据发生联系。关于形参与实参的说明:(1)在定义函数中指定的形参,在未出现函数调用时,它们并不占内存中的存储单元。只有在发生函数调用时,函数max中的形参才被分配内存单元。在调用结束后,形参所占的内存单元也被释放。(2)实参可以是常量、变量或表达式,例如:max(3,a+b);但要求它们有确定的值。在调用时将实参的值赋给形参。(3)在被定义的函数中,必须指定形参的类型。(4)实参与形参的类型应相同或赋值兼容。(5)值传递:实参向形参的数据传递是单向“值传递”,只能由实参传给形参,而不能由形参传回来给实参。在调用函数时,给形参分配存储单元,并将实参对应的值传递给形参,调用结束后,形参单元被释放,实参单元仍保留并维持原值。8.3.2函数的返回值函数的返回值是通过函数调用使主调函数得到的确定值。例如:例8.2中,max(2,3)的值是3,max(5,2)的值是5。赋值语句将这个函数值赋给变量c。说明:(1)函数的返回值是通过函数中的return语句获得的。一个函数中可以有一个以上的return语句,执行到哪一个return语句,哪一个语句起作用。return语句后面的括弧也可以不要例如:“returnz;”等价于“return(z);”return后面的值可以是一个表达式。例如:max(intx,inty){return(xy?x:y);}(2)函数的返回值应当属于某一个确定的类型,在定义函数时指定函数返回值的类型。例如:下面是3个函数的首行:intmax(floatx,floaty)/*函数值为整型*/charletter(charc1,charc2)/*函数值为字符型*/doublemin(intx,inty)/*函数值为双精度型*/(3)在定义函数时指定的函数类型一般应该和return语句中的表达式类型一致。•如果函数值的类型和return语句中表达式的值不一致,则以函数类型为准。•对数值型数据,可以自动进行类型转换。即函数类型决定返回值的类型。(4)对于不带返回值的函数,应当用“void”定义函数为“无类型”(或称“空类型”)。此时在函数体中不得出现return语句。例8.3返回值类型与函数类型不同#includestdio.hintmax(floatx,floaty);voidmain(){floata,b;intc;scanf("%f,%f,",&a,&b);c=max(a,b);printf("Maxis%d\n",c);}intmax(floatx,floaty){floatz;/*z为实型变量*/z=x>y?x∶y;return(z);}运行情况如下:1.5,2.5↙Maxis28.4函数的调用8.4.1函数调用的一般形式函数调用的一般形式为:函数名(实参列表)说明:(1)如果是调用无参函数,则“实参列表”可以没有,但括弧不能省略。(3)如果实参表列包括多个实参,对实参求值的顺序并不是确定的,有的系统按自左至右顺序求实参的值,有的系统则按自右至左顺序。(2)如果实参表列包含多个实参,则各参数间用逗号隔开。实参与形参的个数应相等,类型应匹配。实参与形参按顺序对应,一一传递数据。例8.4实参求值的顺序#includestdio.hvoidmain(){intf(inta,intb);/*函数声明*/inti=2,p;p=f(i,++i);/*函数调用*/printf(%d\n,p);}intf(inta,intb)/*函数定义*/{intc;if(ab)c=1;elseif(a==b)c=0;elsec=-1;return(c);}如果按自左至右顺序求实参的值,则函数调用相当于f(2,3)如果按自左至右顺序求实参的值,则函数调用相当于f(3,3)对于函数调用inti=2,p;p=f(i,++i);8.4.2函数调用的方式1.函数语句把函数调用作为一个语句。这时不要求函数带回值,只要求函数完成一定的操作。2.函数表达式函数出现在一个表达式中,这种表达式称为函数表达式。这时要求函数带回一个确定的值以参加表达式的运算。例如:c=2*max(a,b);按函数在程序中出现的位置来分,可以有以下三种函数调用方式:3.函数参数函数调用作为一个函数的实参。例如:m=max(a,max(b,c));其中max(b,c)是一次函数调用,它的值作为max另一次调用的实参。m的值是a、b、c三者中的最大者。8.4.3对被调用函数的声明和函数原型1.首先被调用的函数必须是已经存在的函数(是库函数或用户自己定义的函数)。但光有这一条件还不够。3.如果使用用户自己定义的函数,而该函数的位置在调用它的函数(即主调函数)的后面,应该在主调函数中对被调用的函数作声明。2.如果使用库函数,还应该在本文件开头用#include命令将调用有关库函数时所需用到的信息“包含”到本文件中来。函数声明的一般形式为:1.函数类型函数名(参数类型1,参数类型2……);2.函数类型函数名(参数类型1,参数名1,参数类型2,参数名2……);声明的作用是把函数名、函数参数的个数和参数类型等信息通知编译系统,以便在遇到函数调用时,编译系统能正确识别函数并检查调用是否合法。注意:函数的“定义”和“声明”的区别:函数的定义是指对函数功能的确立,包括指定函数名,函数值类型、形参及其类型、函数体等,它是一个完整的、独立的函数单位。函数的声明的作用则是把函数的名字、函数类型以及形参的类型、个数和顺序通知编译系统,以便在调用该函数时系统按此进行对照检查。例8.5对被调用的函数作声明#includestdio.hfloatadd(floatx,floaty);/*对被调用函数add的声明*/voidmain(){floata,b,c;scanf("%f,%f",&a,&b);c=add(a,b);printf("sumis%f\n",c);}floatadd(floatx,floaty)/*函数首部*/{floatz;/*函数体*/z=x+y;return(z);}例8.5对被调用的函数作声明#includestdio.hfloatadd(floatx,floaty)/*函数首部*/{floatz;/*函数体*/z=x+y;return(z);}voidmain(){floata,b,c;scanf("%f,%f",&a,&b);c=add(a,b);printf("sumis%f\n",c);}8.5函数的嵌套调用嵌套定义就是在定义一个函数时,其函数体内又包含另一个函数的完整定义。例8.6用弦截法求方程f(x)=x3-5x2+16x-80=0的根8.6函数的递归调用在调用一个函数的过程中又出现直接或间接地调用该函数本身,称为函数的递归调用。C语言的特点之一就在于允许函数的递归调用。例如:intf(intx){inty,z;z=f(y);return(2*z);}例8.7:有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)=10用