《C语言与程序设计教程》课件第5章.ppt
- 【下载声明】
1. 本站全部试题类文档,若标题没写含答案,则无答案;标题注明含答案的文档,主观题也可能无答案。请谨慎下单,一旦售出,不予退换。
2. 本站全部PPT文档均不含视频和音频,PPT中出现的音频或视频标识(或文字)仅表示流程,实际无音频或视频文件。请谨慎下单,一旦售出,不予退换。
3. 本页资料《《C语言与程序设计教程》课件第5章.ppt》由用户(momomo)主动上传,其收益全归该用户。163文库仅提供信息存储空间,仅对该用户上传内容的表现方式做保护处理,对上传内容本身不做任何修改或编辑。 若此文所含内容侵犯了您的版权或隐私,请立即通知163文库(点击联系客服),我们立即给予删除!
4. 请根据预览情况,自愿下载本文。本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。
5. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007及以上版本和PDF阅读器,压缩文件请下载最新的WinRAR软件解压。
- 配套讲稿:
如PPT文件的首页显示word图标,表示该PPT已包含配套word讲稿。双击word图标可打开word文档。
- 特殊限制:
部分文档作品中含有的国旗、国徽等图片,仅作为作品整体效果示例展示,禁止商用。设计者仅对作品中独创性部分享有著作权。
- 关 键 词:
- C语言与程序设计教程 语言 程序设计 教程 课件
- 资源描述:
-
1、第第5章章 函数函数 5.1 函数的概念及特点 5.2 函数的定义和调用 5.3 变量的作用域 5.4 函数的嵌套调用与递归调用5.5 典型例题精讲*5.6 递归转化为非递归研究 5.1 函数的概念及特点函数的概念及特点 5.1.1 函数的概念函数的概念对于一些规模较大而又比较复杂的问题,解决的方法往往是把它们分解成若干个较为简单和基本的问题进行求解。这在程序设计中表现为:将一个大程序分解为若干个相对独立且较为简单的子程序,大程序通过调用这些子程序来完成预定的任务。子程序的引入不仅可以较容易地解决一些复杂问题,而且更重要的是使程序有了一个层次分明的结构。另外,程序中反复出现的相同程序段也可以采
2、用子程序的形式独立出来,以供程序随时调用。子程序结构不但增强了程序的可读性,同时也使一个复杂程序的编写、调试、维护和扩充都变得更加方便和容易。主程序与子程序间的调用关系如图5-1所示,主程序顺序执行到调用子程序语句时,就转去执行子程序;当子程序执行结束时,再返回到主程序中调用子程序语句的下一条语句继续向下执行。图5-1 主程序与子程序调用关系示意图在C语言中,子程序的作用是由函数来完成的。一个C程序可由一个主函数(即main函数)和若干个其他函数构成。主函数main可以调用其他函数,其他函数之间也可以相互调用,同一个函数可以被一个或多个函数调用任意多次。C程序的执行总是从主函数main开始,其
3、间可以调用其他函数,但最终还是回到main函数并在main函数中结束整个程序的运行。5.1.2 函数的分类函数的分类从函数定义角度看,函数可分为库函数和用户自定义函数两种。1.库函数库函数由C语言系统直接提供且不必在程序中作类型说明,只需在程序前包含该函数原型的头文件即可。前面各章例题中反复用到的printf、scanf、getchar、putchar、gets、puts、strcat等函数均属此类。例如,要使用数学函数,则需用“#include”将数学头文件包含到程序中;要使用字符串处理函数则需用“#include”将字符串处理头文件包含到程序中。注意,C程序中调用库函数时需分两步实现。第一
4、步:在程序开始处使用include命令指出库函数的相关定义和说明。而include命令必须以“#”开头,系统提供的头文件以“.h”作为文件后缀,文件名用一对尖括号“”或一对双引号“”括起来。由于以#include开头的命令行不是C语言语句,故该命令行末尾不加分号“;”。第二步:在程序中需要调用这个库函数的地方调用此库函数,其形式为:库函数名(参数表)常见的库函数如下:(1)输入输出函数(头文件为stdio.h):用于完成输入输出功能。(2)字符串函数(头文件为string.h):用于字符串操作和处理。(3)数学函数(头文件为math.h):用于数学函数计算。(4)内存管理函数(头文件为stdl
5、ib.h):用于内存管理。(5)日期和时间函数(头文件为time.h):用于日期、时间的转换操作。(6)接口函数(头文件为dos.h):用于与DOS、BIOS和硬件的接口。C语言常用的标准库函数及调用方式参见附录3。2.用户自定义函数用户自定义函数是指用户按需要自行定义和编写的函数。虽然C语言的标准库函数为用户提供了丰富的函数,但还远远不能满足用户实际编程的需要。因此,大量的函数还需用户自行定义。如何定义一个函数以及如何正确调用一个函数是本章讨论的重点。5.2 函数的定义和调用函数的定义和调用 5.2.1 函数的定义函数的定义在C语言中,所有自定义的函数都必须遵循“先定义,后使用”的原则,在使
6、用之前先进行定义。并且,所有的函数定义包括主函数main都是相互平行和独立的,不允许出现嵌套定义。函数定义的形式为:类型标识符 函数名(形式参数表)函数体其中第一行称为函数首部,它定义了函数返回值的类型、函数的名字以及调用该函数时需要给出的参数个数及参数的类型。需要注意的是,函数名为有效的标识符,不能与其他变量名、数组名相同。此外,如果所定义的函数不需要参数,则形式参数表可以省略,但括号“()”不能省略。用花括号“”括起来的部分称为函数体,它包括函数的说明部分和执行部分。说明部分用于对函数内部使用的变量进行定义,也即在此定义的变量仅在该函数内部有效;执行部分是函数的主体,它具体描述该函数所应实
7、现的功能。根据完成功能和调用方式的不同,C语言的函数又分为有返回值函数和无返回值函数两种类型。1无返回值函数无返回值函数其典型标志是函数返回值的类型为void,这种函数只是完成某种指定的操作,并且调用该函数通过一个函数调用语句实现。例如:#includevoid f(int x);/*自定义函数f的声明语句*/void main()int a=5;f(a);void f(int x)/*自定义函数f*/printf(%dn,x);在此,自定义函数f只是完成将主函数main传递过来的a值(已保存于形参x中)进行输出这一功能,而主函数main调用函数f也是通过一个函数调用语句“f(a);”实现的。
8、2有返回值函数有返回值函数其典型标志是函数返回值的类型为int、char、float和指针类型(见第6章)等,这种函数实现的操作主要是为了获得一个计算的结果值,而这个结果值最终必须通过return语句返回给调用者,并且这个返回值的类型就是由函数首部这个类型标识符来指定的。对于该类函数调用可以出现在表达式中,即出现在赋值号“=”右边或printf输出语句中。例如:#includeint f(int x);/*自定义函数f的声明语句*/void main()int a=5;printf(%dn,f(a);int f(int x)/*自定义函数f*/return(x*x);在此,自定义函数f完成将主
9、函数main传递过来的a值(已保存于形参x中)平方后再通过return语句传回给main函数,而主函数main调用函数f是以表达式的形式出现在printf语句中。需要注意的是,这类函数在定义时通常要有将函数计算结果返回给调用者的return语句,并且通常是在表达式中调用这类函数。在PASCAL等高级语言中,像C语言中这种无返回值函数被称之为过程,而有返回值函数才称之为函数。5.2.2 函数的调用和返回值函数的调用和返回值1.函数的调用函数虽然不可以嵌套定义,但可以嵌套调用,也允许函数之间相互调用。通常我们把调用者称为主调函数而把被调用者称为被调函数。函数还可以自己调用自己,被称为递归调用。在大
10、多数情况下,函数调用时主调函数与被调函数之间存在着数据传递(也称参数传递)。需要说明的是,在定义函数时函数名后面括号“()”中的变量名称为“形式参数”(简称“形参”);在主调函数中调用一个函数时,该函数名后面括号“()”中的参数称为“实际参数”(简称“实参”)。形式参数本身只具有形式上的意义,仅当给形式参数赋予了调用的实际参数后才有确切的内容。形式参数与程序中的变量定义类似,也有值的类型。因此在函数定义时必须指定每个形式参数的类型。在函数调用时,参数传递的方式是将主调函数中的实参(可以是常量、变量或表达式)传递给被调函数中的形参,这是一种值的传递。参数传递的原则是:形参与实参的排列顺序必须一致
11、,类型必须相同或赋值相容,参数的个数也必须相同。需要注意的是,在定义函数时指定的形参,在未出现函数调用时,它们并不占用内存单元,只有在发生函数调用时,这些形参才被分配内存单元并接受所对应的实参值。当函数调用结束时,形参占用的内存单元被系统收回而不再存在;函数内定义的变量(称为局部变量)也是如此。当再次出现函数调用时,则重复上述的分配和回收过程。函数调用的过程是:当主调函数调用被调函数时,先为被调函数的形参开辟对应的内存单元,然后计算出实参表达式的值并送入对应的形参内存单元;至此参数传递完成,形参与实参之间就不再发生联系,形参的任何变化也不影响实参。这种参数传递如同接力比赛的交接棒一样,当交完棒
12、后,交棒者和接棒者就再无关系了。当函数执行结束时,形参因其内存单元被系统收回而不存在,故形参仅在函数中有意义。因此,可以把形参看做是函数中的局部变量,而实参传递给形参就相当于给形参赋初值。例如:#includeint sum(int x,int y);void main()int a=1,b=2,c;c=sum(a,b);printf(c=%dn,c);int sum(int x,int y)int z;z=x+y;return z;在此,主调函数main将两个整数a和b的值分别传递给被调函数sum的形参x和y。而函数sum则实现对x和y两数求和的功能,并通过变量z将求和的结果返回给主调函数m
13、ain的变量c,最终通过函数printf输出这个c值。2.函数的返回值函数的返回值是指调用被调函数后,执行函数体中的语句得到所需的计算结果,并将这个结果值通过return语句返回给主调函数,这个值就是函数的返回值。在定义带返回值的函数时,必须先定义函数的类型。函数类型决定了函数返回值的类型,这里要注意以下两点:(1)函数返回值的类型和函数的类型应保持一致。如果两者不一致则以函数定义中的函数类型为准,将函数的返回值自动转换为函数定义的类型。(2)函数返回值通常由return语句返回给主调函数。return语句的功能是计算表达式的值并返回给主调函数,只要执行return语句,则结束被调函数的执行并
14、将返回值(如果有的话)返回给主调函数。在一个函数中可以有多条return语句,但每次函数调用只能有一条return语句被执行(当执行该条return语句后即返回到主调函数,故其余return语句无法执行),所以一个函数只能返回一个函数值。return语句的格式分为下面三种:(1)return表达式;(2)return(表达式);(3)return;其中,(1)、(2)两种格式的功能是一样的,它们都能实现:将返回值返回给主调函数;终止被调函数的执行;返回到主调函数。第(3)种格式与第(1)、(2)种格式的差别在于无返回值返回给主调函数。注意,VC+6.0有返回值函数使用格式(1)、(2),而无返
15、回值(void开头)函数通常不使用return语句,如果使用也只能用格式(3)。例如:#includeint max(int x,int y);void main()int a=5,b=6,c;c=max(a,b);printf(%dn,c);int max(int x,int y)if(xy)return x;elsereturn y;此外,要注意的是,定义为void类型的函数中是没有return语句的,这类函数只是完成需要的操作而无返回值。5.2.3 函数执行的分析方法函数执行的分析方法为了能够形式化地描述函数调用执行的全部过程,我们采用称之为动态图的方法来对程序的执行进行描述,即记录主函
16、数和被调函数的调用、运行及撤消各个阶段的状态,以及程序运行期间所有变量和函数形参的变化过程。动态图规则如下:(1)动态图纵向描述主函数和其他函数各层之间的调用关系,横向由左至右按执行的时间顺序记录主函数和其他函数中各变量及形参值的变化情况。(2)函数的形参均看做是带初值的局部变量。其后,形参就作为局部变量参与函数中的操作。(3)主函数、其他函数按运行中的调用关系自上而下分层,各层说明的变量包括形参都依次列于该层首列,各变量值的变化情况按时间顺序记录在与该变量对应的同一行上。例5.1 用动态图分析下面程序的执行过程。#includeint max(int x,int y);void main()
17、int a=5,b=6,c;c=max(a,b);printf(%dn,c);int max(int x,int y)int z;z=xy?x:y;return(z);解 函数max的形参为x、y,在函数调用时y、x分别接受了实参b、a的值6和5。注意,在VC+6.0中参数传递是由右左进行的(参见例5.2)。此外,函数max还定义了一个局部变量z来保存x和y值中较大的那一个值,最终将这个z值返回给主调函数main。图5-2以动态图的方式描述了整个程序的执行情况,其中包括函数max因调用而创建、运行及运行结束后撤消的全过程。由动态图可知该程序的输出结果为6。图5-2 程序运行和函数调用动态图例5
18、.2 用动态图分析下面程序的执行过程。#includeint add(int x,int y);void main()int i=1,sum;sum=add(i,+i);printf(%dn,sum);int add(int x,int y)int z;z=x+y;return(z);运行结果:4 解 主调函数main调用被调函数add的实参依次为i和+i,根据参数传递由右向左的原则,则先执行+i,即i值变为2后传递给形参y,接着再将i值(已为2)传给x。图5-3(a)中的动态图描述了整个程序的执行情况,因此最终的输出结果是4而不是3,程序上机执行也证实了这种结果。如果调用函数的语句改为sum
19、=add(+i,i);,则最终输出结果是3而不是4,这也证实了参数传递是由右向左进行的。此时程序的执行情况见图5-3(b)的动态图描述。图5-3 程序运行和函数调用动态图5.2.4 函数的声明如果被调函数定义在主调函数之前,则满足“先定义,后使用”的原则,主调函数可在函数体内直接调用被调函数;如果被调函数定义在主调函数之后,那么主调函数在函数体内直接调用被调函数则成了“先使用,后定义”,即违背了“先定义,后使用”的原则。在有些情况下不可避免的会出现“先使用,后定义”这种情况:如两个函数相互调用时,则必然有一个函数调用会出现“先使用,后定义”的问题。为了解决这种矛盾,即被调用函数定义在主调函数之
20、后时,则必须在程序中予以声明。声明的方法是将后定义的被调函数其函数首部(即该函数定义的第一行)再加上一个分号“;”放置于主调函数之前或主调函数中的说明部分。这样,当在主调函数中使用被调函数时由于其前面已有了该函数的“声明”,则认为被调函数已经定义过了,即满足了“先定义,后使用”的原则。例5.3 分析下面程序进行两数交换的执行过程。#includevoid main()void swap(int a,int b);int x=8,y=10;printf(x=%dty=%dn,x,y);swap(x,y);printf(Swapped:n);printf(x=%dty=%dn,x,y);void
21、swap(int a,int b)int temp;temp=a;a=b;b=temp;运行结果:x=8 y=10Swapped:x=8 y=10 解 本例中,被调用函数swap定义于主调函数之后,因此在主调函数main调用被调函数swap的调用语句“swap(x,y);”之前必须先对函数swap进行声明。在程序中,这个声明语句位于主函数main之前(也可放在主调函数main中“swap(x,y);”语句之前的任何位置)。此外,由图5-4可以看出,参数传递只是一种单向传递,即由实参传给形参。而执行被调函数所引起形参值的任何变化都不会影响到实参,故执行完函数swap后返回到主函数main时,其x
22、、y值均没有发生改变。图5-4 例5-3的程序动态图从例5.3可以看出,函数的声明与函数定义中的函数首部只差一个分号“;”,因此可以简单地照抄已定义的函数首部,再加上一个分号“;”,就形成了该函数的“声明”语句。此外,也可以省略函数首部中的形参名,仅保留形参的类型。如上例函数swap的声明可写为:void swap(int,int);此外,例5.3也可以按“先定义,后使用”方式书写:#includevoid swap(int a,int b)int temp;temp=a;a=b;b=temp;void main()int x=8,y=10;printf(x=%dty=%dn,x,y);swa
23、p(x,y);printf(Swapped:n);printf(x=%dty=%dn,x,y);5.3 变量的作用域变量的作用域 当程序中存在两个以上的函数时,就会出现变量的作用域问题。变量的作用域是指变量在程序中可以被使用的范围。在C语言中,根据变量的作用域可以将变量分为局部变量和全局变量。变量说明位置不同,其作用域也不同。5.3.1 全局变量与局部变量全局变量与局部变量在函数内部定义的变量或者在复合语句内部定义的变量,称之为局部变量。它只在该函数范围内或该复合语句内有效,离开这个范围变量就无效了。由于函数中的形参只在该函数内有效,离开该函数就不存在。所以也可以将形参看做接受了实参初值的局部
24、变量(局限于该函数内)。全局变量是指在程序中任何函数之外定义的变量,这种变量的使用不局限于某一个函数,而是从定义处开始位于其后的函数都可使用,也就是由定义处开始至程序结束都有效(即满足“先定义后使用”原则)。如果在函数内有与全局变量同名的局部变量存在,那么究竟哪个变量起作用呢?我们将全局变量和局部变量的适用范围归纳如下:(1)如果全局变量与局部变量不同名,则全局变量适用于由定义处开始至程序结束这个范围,而函数说明的变量及形参则仅局限于在该函数内使用。(2)若全局变量与局部变量同名,则在说明该局部变量的函数之外,同名的全局变量起作用(当然应属于由全局变量定义处开始至程序结束这一范围),而同名的局
25、部变量则不存在;在函数内,则同名的局部变量起作用,而同名的全局变量暂时不起作用。注意,C语言中规定主函数main与其他函数的地位是相同的,它们是相互平行的。主函数中定义的变量只能在主函数中使用而不能在其他函数中使用,主函数中也不能使用其他函数中定义的变量。在程序中定义全局变量的目的是增加函数间数据传递的通道以及实现数据共享:(1)如果在程序的开始处定义了全局变量,则这个程序中所有函数都能引用全局变量的值。也即,如果在一个函数中改变了全局变量的值,则其他函数就可以访问这个改变了值的全局变量,这样相当于多了一个数据传递的通道。(2)由于函数的调用只能带回一个返回值,则在需要带回两个以上的结果值时就
展开阅读全文