【文档说明】C语言函数解析课件.ppt,共(66)页,3.024 MB,由小橙橙上传
转载请保留链接:https://www.ichengzhen.cn/view-44564.html
以下为本文档部分文字说明:
第7章函数哈尔滨工业大学计算机科学与技术学院苏小红sxh@hit.edu.cn2022/11/242/66本章学习内容函数定义、函数调用、函数原型、函数的参数传递与返回值递归函数和函数的递归调用函数封装,函数复用,函数设计的基本原则,程序的健壮性变量的作用域与存储类型,全局变量、自动变量
、静态变量、寄存器变量“自顶向下、逐步求精”的模块化程序设计方法2022/11/243/66数学中的函数()yfx自变量因变量函数名程序设计中的函数程序设计中的函数不局限于计算–计算类,如打印阶乘表的程序……–判断推理类,如排序、查找……2
022/11/244/66问题的提出读多少行的程序能让你不头疼?假如系统提供的函数printf()由10行代码替换,那么你编过的程序会成什么样子?–实际上一个printf()有上千行代码main()中能放多少行代码?如果所有代码都在mai
n()中,怎么团队合作?如果代码都在一个文件中,怎么团队合作?2022/11/245/66问题的提出《三国演义》中有这样一段描写:–懿问曰:“孔明寝食及事之烦简若何?”使者曰:“丞相夙兴夜寐,罚二十以上皆亲览焉。所啖之食,
日不过数升。”懿顾谓诸将曰:“孔明食少事烦,其能久乎?”–此话音落不久,诸葛亮果然病故于五丈原。“事无巨细”,“事必躬亲”–管理学的观点是极其排斥这种做法的,认为工作必须分工,各司其职–其中的思想,在程序设计里也
适用2022/11/246/667.1分而治之与信息隐藏分而治之(DivideandConquer,Wirth,1971)–函数把较大的任务分解成若干个较小的任务,并提炼出公用任务信息隐藏(InformationHiding,Parnas,1972)–设计得当的函数可
把具体操作细节对外界隐藏起来,从而使整个程序结构清楚–使用函数时,不用知道函数内部是如何运作的,只按照我们的需要和它的参数形式调用它即可2022/11/247/66程序设计的艺术算法设计艺术–程序的灵魂–DonaldE.Kn
uth,“TheArtofComputerProgramming”,清华大学出版社(英),国防工业出版社(中)结构设计艺术–程序的肉体–模块化(Parnas,1972)结构化(Structural)面向对象(Object-Oriented)面向组件(Component-Oriented)
面向智能体(Agent-Oriented)……2022/11/248/66函数是C语言中模块化编程的最小单位–可以把每个函数看做一个模块(Module)如把编程比做制造一台机器,函数就好比其零部件–可将这些“零部件”单独设计、调试、测试好,用时拿出来装配,再总体调试。–这些“零部件”可
以是自己设计制造/别人设计制造/现成的标准产品7.2函数的定义MoeCurlyLarry2022/11/249/667.2函数的定义若干相关的函数可以合并成一个“模块”一个C程序由一个或多个源程序文件组成一个源程序文件由一个或多个函数组成2022/11/2410/667.2.1函数
的分类函数生来都是平等的,互相独立的,没有高低贵贱和从属之分–main()稍微特殊一点点–C程序的执行从main()函数开始–调用其他函数后流程回到main()函数–在main()函数中结束整个程序运行2022/11/2411/667.2.1函数的分类标准库函数–ANSI/IS
OC定义的标准库函数符合标准的C语言编译器必须提供这些函数函数的行为也要符合ANSI/ISOC的定义–第三方库函数由其他厂商自行开发的C语言函数库不在标准范围内,能扩充C语言的功能(图形、网络、数据库等)自定义函数–自己定义的函数包装后,也可成为函数库,供别人使用2
022/11/2412/667.2.2函数的定义(FunctionDefinition)类型函数名(类型参数1,类型参数2,……){声明语句序列可执行语句序列return表达式;}返回值类型函数名标识符,说明
运算规则参数表相当于运算的操作数返回运算的结果函数出口2022/11/2413/66类型函数名(类型参数1,类型参数2,……){声明语句序列可执行语句序列return表达式;}函数体的定界符参数表里的变量(叫形式参数,FormalParameter)也是内部变量
函数体7.2.2函数的定义(FunctionDefinition)2022/11/2414/66void函数名(void){声明语句序列可执行语句序列return;}函数无返回值,用void定义返回值类型用void定义参数,表示没有参数return语句后无需任何表达式7.2.
2函数的定义(FunctionDefinition)2022/11/2415/66【例7.1a】计算整数n的阶乘n!/*函数功能:用迭代法计算n!函数入口参数:整型变量n表示阶乘的阶数函数返回值:返回n!的值*
/longFact(intn)/*函数定义*/{inti;longresult=1;for(i=2;i<=n;i++){result*=i;}returnresult;}返回值类型函数名说明函数的功能返回值作为函数调用表达式的值形参表,函数入口函数内
部可以定义只能自己使用的变量,称内部变量2022/11/2416/66函数名(表达式1,表达式2,……);实际参数(ActualArgument)–函数调用(FounctionCall)时提供的表达式有返回值时–放到一个数值表达式中c=max(a,b)
;–作为另一个函数调用的参数c=max(max(a,b),c);printf("%d\n",max(a,b));无返回值时–函数调用表达式display(a,b);返回值=函数名(实参表列);函数名(实参
表列);7.3向函数传递简单变量的值和从函数返回值2022/11/2417/66函数的参数传递实参和形参必须匹配–数目一致,类型一一对应(否则会发生自动类型转换)【例7.1】2022/11/2418/667.3.2函数原型(FunctionPrototyp
e)在调用函数前先声明其返回值类型、函数名和参数函数原型有助于编译器对函数参数类型的匹配检查末尾有一个分号,声明时不要省略形参和返回值的类型【例7.1】2022/11/2419/66函数定义与函数声明的区别函数定义–指函数功能的确立–指定函数名、函数类型、形
参及类型、函数体等–是完整独立的单位函数声明–是对函数名、返回值类型、形参类型的说明–不包括函数体–是一条语句,以分号结束,只起一个声明作用2022/11/2420/667.3.3函数封装与防御性程序设计函数封装(Enca
psulation)使得外界对函数的影响仅限于入口参数,而函数对外界的影响仅限于一个返回值和数组、指针类型的参数【例7.1】Why?传入负数实参会怎样?2022/11/2421/66防御性程序设计(DefensiveProgramming)如何使函数具有遇
到不正确使用或非法数据输入时避免出错的能力,增强程序的健壮性?–在函数的入口处,检查输入参数的合法性【例7.2】计算整数n的阶乘n!2022/11/2422/66如何使函数具有遇到不正确使用或非法数据输入时避免出错的能力,增强程序的健壮性?–在函数的入口处,检查输入参
数的合法性防御性程序设计(DefensiveProgramming)【例7.2】计算整数n的阶乘n!2022/11/2423/66主函数如何修改?–增加对函数返回值的检验防御性程序设计(DefensiveProgramming)【例7.3】计算整数n的阶乘n!2022/11/2424/66传入
负数的实参时Fact()会返回-1吗?–存在死代码的原因何在?防御性程序设计(DefensiveProgramming)【例7.3】计算整数n的阶乘n!2022/11/2425/66如何修改程序去除冗余代码?如何保证不会传入负数实参?防御性程序设计(DefensiveProgr
amming)【例7.2】计算整数n的阶乘n!2022/11/2426/66【例7.4】编写计算组合数的程序函数复用2022/11/2427/667.3.4函数设计的基本原则信息隐藏1函数规模要小2函数功能要单一3函数接口定义要清楚入口参数有效性检
查敏感操作前的检查调用成功与否的检查2022/11/2428/66函数的嵌套调用嵌套调用–在调用一个函数的过程中,又调用另一个函数C语言规定函数不能嵌套定义,但可以嵌套调用–函数是相互平行的main(){……a();}a函数{b();…return;}b函数{……return;
}①③④⑤⑥⑦②2022/11/2429/667.4函数的递归调用和递归函数如果一个对象部分地由它自己组成或按它自己定义,则我们称它是递归(Recursive)的。‒生活中,字典就是一个递归问题的典型实例‒字典中的任何一个词汇都是由“其他词汇”解释或定义的,但是“其他
词汇”在被定义或解释时又会间接或直接地用到那些由它们定义的词递归方法的基本原理–将复杂问题逐步化简,最终转化为一个最简单的问题–最简单问题的解决就意味着整个问题的解决2022/11/2430/66递归函数(Recur
siveFunction)longfact(intn){if(n<0)return-1;elseif(n==0||n==1)return1;elsereturnn*fact(n-1);}【例7.5】计算n!=n*(n-1)*(n-2)*……*
1函数直接或间接调用自己,称为递归调用(RecursiveCall)2022/11/2431/66递归函数(RecursiveFunction)unsignedlongfact(unsignedintn){if(n==0||
n==1)return1;elsereturnn*fact(n-1);}基线情况(basecase)一般情况(generalcase)无需考虑n<0了【例7.5】计算n!=n*(n-1)*(n-2)*…*12022/11/2432/66递归函数(RecursiveFunc
tion)递归调用应该能够在有限次数内终止递归–递归调用若不加以限制,将无限循环调用–必须在函数内部加控制语句,仅当满足一定条件时,递归终止,称为条件递归任何一个递归调用程序必须包括两部分–递归循环继续的过程–递归调用结束的过程if(递归终止条件成立)return递归公式的初值;el
sereturn递归函数调用返回的结果值;2022/11/2433/66n!=n×(n-1)!(n-1)!=(n-1)×(n-2)!(n-2)!…(n-3)!5!…4!=4×3!3!=3×2!2!=2×1!1!=1
回推过程递推过程每个递归函数必须至少有一个基线条件一般情况必须最终能简化为基线条件递归层数太多易导致栈空间溢出后果很严重,程序被异常中止fact(5)=5*fact(4)=120fact(4)=4*fact(3)=24fact(3)=3*fact(2)=6fact(2)=2*
fact(1)=2fact(1)=1mainfact(5)fact(4)fact(3)fact(2)fact(1)2022/11/2434/66递归与迭代用迭代(即循环)方法编写的阶乘函数unsignedlongFact(unsignedintn)
{unsignedlongresult=1;unsignedinti;for(i=1;i<=n;i++)result*=i;returnresult;}递归程序遵循了数学中对阶乘的定义因此递归方法编写程序具有更清晰、可读性更好的优点2022/11/2435/66递归与迭代1,1,2,3,5
,8,......110)2()1(10)(nnnnfibnfibnfiblongFib(intn){longf;if(n==0)f=0;elseif(n==1)f=1;elsef=Fib(n-1)+Fib(n-2);re
turnf;}154323221101010计算fib(5)共需15次fib调用【例7.6】计算Fibonacci数列2022/11/2436/66递归与迭代优点:–从编程角度来看,比较直观、精炼,逻
辑清楚–符合人的思维习惯,逼近数学公式的表示–尤其适合非数值计算领域hanoi塔,骑士游历、八皇后问题(回溯法)缺点:–增加了函数调用的开销,每次调用都需要进行参数传递、现场保护等–耗费更多的时间和栈
空间–应尽量用迭代形式替代递归形式2022/11/2437/667.5变量的作用域和存储类型7.5.1变量的作用域(Scope)指在源程序中定义变量的位臵及其能被读写访问的范围分为–局部变量(LocalVariable)–全局变
量(GlobalVariable)2022/11/2438/66局部变量(LocalVariable)在语句块内定义的变量–形参也是局部变量特点–生存期是该语句块,进入语句块时获得内存,仅能由语句块内语句访问,退出语句块时释放内存,不再有效–定义时不会自动初始化,
除非程序员指定初值–并列语句块各自定义的同名变量互不干扰形参和实参可以同名2022/11/2439/66全局变量(GlobalVariable)在所有函数之外定义的变量–生存期是整个程序,从程序运行起占据内存,程序运
行过程中可随时访问,程序退出时释放内存–有效范围是从定义变量的位臵开始到本程序结束2022/11/2440/66全局变量(GlobalVariable)【例7.7】打印计算Fibonacci数列每一项时所需的递归调用次数全局变量使函数间的数据交换更容
易,更高效,但建议尽量少用,因为谁都可改写它,所以很难确定是谁改写了它全局变量2022/11/2441/667.5.2变量的存储类型(StorageClass)指数据在内存中存储的方式–即编译器为变量分配内存的方式,它决定变量的生存期存储类型数据类型变量名;C程序
的存储类别–auto型(自动变量)–static型(静态变量)–extern型(外部变量)–register型(寄存器变量)2022/11/2442/66静态存储区中的变量:与程序“共存亡”动态存储区中的变量:与程序块“共存亡”寄存器中的变量:同动态存储
区变量的生存期(Lifetime)决定何时“生”,何时“灭”7.5.2变量的存储类型(StorageClass)2022/11/2443/66auto数据类型变量名;auto体现在–进入语句块时自动申请内存,退出时自动释放内存–动态局部变量,缺省
的存储类型静态变量static数据类型变量名;–staticstorageclassforlocalvariables(declaredinsideablockorfunction)-thelifetimeoftheentirepr
ogram–生存期为整个程序运行期间自动变量和静态变量2022/11/2444/66自动变量和静态变量【例7.9】自动变量不初始化时,值是随机值,静态局部变量和全局变量自动初始化为0值2022/11/2445/66【例7.1
0】利用静态变量计算整数n的阶乘n!自动变量和静态变量静态变量仅初始化一次,变量的值可保存到下次进入函数,使函数具有记忆功能2022/11/2446/66寄存器变量寄存器–CPU内部容量有限、但速度极
快的存储器register类型名变量名;使用频率比较高的变量声明为register,可使程序更小、执行速度更快–现代编译器有能力自动把普通变量优化为寄存器变量,并且可以忽略用户的指定–所以一般无须特别声明变量
为register2022/11/2447/66全局变量静态外部变量(只限本文件使用)外部变量(非静态外部变量,允许其他文件引用)局部变量自动变量,(离开函数,值就消失)寄存器变量(离开函数,值就消失)定义点之前使用,需用extern声明静态局部变量(离开函数,值仍保留)
动态局部变量7.5变量的作用域和存储类型2022/11/2448/667.6模块化程序设计模块各司其职–每个模块只负责一件事情,它可以更专心–便于进行单个模块的设计、开发、调试、测试和维护等工作–一个模块
一个模块地完成,最后再将它们集成开发人员各司其职–按模块分配任务,职责明确–并行开发,缩短开发时间什么时候需要模块化?–某一功能,如果重复实现3遍以上,即应考虑模块化,将它写成通用函数,并向小组成员发布2022/11/2
449/667.6模块化程序设计模块化的优点——复用–构建新的软件系统可以不必每次从零做起,直接使用已有的经过反复验证的软构件,组装或加以合理修改后成为新的系统,提高软件生产率和程序质量在其他函数的基
础上构造程序–拿来拿去主义,指尽可能复用其他人现成的模块不是人类懒惰的表现,而是智慧的表现–一般要靠日积月累才能建立可以被复用的软件库前期投入多,缺乏近期效益,大部分公司都注重近期效益,是为了生存,所以软件复用对公司来说不是最高优先级2022/
11/2450/667.6模块化程序设计功能分解–自顶向下、逐步求精的过程模块分解的基本原则–保证模块的相对独立性——高聚合、低耦合–模块的实现细节对外不可见——信息隐藏外部:关心做什么;内部:关心怎么做设计好模块接口–接口指罗列出一个模块的所有的与外部打交道的变
量等–定义好后不要轻易改动–在模块开头(文件的开头)进行函数声明2022/11/2451/667.6模块化程序设计逐步求精(StepwiseRefinement)–由不断的自底向上修正所补充的自顶向下(Top-down)的程序设计方法2022/11/24
52/66【例7.11】用函数完成猜数游戏猜多个数,10次猜不对就猜下一个数模块分解过程开始结束初始化退出处理主功能为程序运行所做的准备工作在退出前要做的事情,如打印结果、资源释放等自底向上自顶向下的模块化程序设
计2022/11/2453/66【例7.11】用函数完成猜数游戏开始结束生成数字猜数字2022/11/2454/66【例7.11】用函数完成猜数游戏开始结束生成数字猜数字是否继续?NY2022/11/2455/6
6【例7.11】用函数完成猜数游戏开始结束猜得对吗?NY提示大小次数<10?输入数字NY处理用户输入,判断是否有输入错误,是否在合法的数值范围内2022/11/2456/66【例7.11】用函数完成猜数游戏2022/11/2457/66【例7.11】用函数完成猜数游戏2022/11/
2458/66【例7.11】用函数完成猜数游戏2022/11/2459/66【例7.11】用函数完成猜数游戏intMakeNumber(void){intnumber;number=(rand()%(MAX_NUMBER-MIN_NUMBER+1))+MIN_NUMBER;assert(numbe
r>=MIN_NUMBER&&number<=MAX_NUMBER);returnnumber;}使用断言(Assert)防止某些参数获得非法值,在程序调试和测试时发现错误2022/11/2460/
66#include<assert.h>voidassert(intexpression);–expression为真,无声无息;为假,中断程序断言仅用于调试程序,不能作为程序的功能断言用来测试某种不可能发生的状况确实不会发生–Debug版有效–Releas
e版失效考虑使用断言的几种情况–检查程序中的各种假设的正确性–证实或测试某种不可能发生的状况确实不会发生2022/11/2461/66程序版式缩进(Indent)—保证代码整洁、层次清晰的主要手段良好风格的程序应严格采用梯形层次对应好各层次intIs
Prime(intn){intk,i;k=sqrt((double)n);for(i=2;i<=k;i++){if(n%i==0)return0;}return1;}#include<math.h>main(){inti;for(i=2;i<100;i++
){if(IsPrime(i))printf("%d\t",i);}}2022/11/2462/66程序版式现在的许多开发环境、编辑软件都支持自动缩进–根据用户代码的输入,智能判断应该缩进还是反缩进,替用户完成调整缩
进的工作VC中有自动整理格式功能–只要选取需要的代码,按ALT+F8就能自动整理成微软的cpp文件格式2022/11/2463/66命名规则在Linux/UNIX平台–习惯用function_name本书采用Windows风格函
数名命名–用大写字母开头、大小写混排的单词组合而成–FunctionName变量名形式–“名词”或者“形容词+名词”–如oldValue与newValue等函数名形式–“动词”或者“动词+名词”(动宾词组)–如GetMax()等2022/11
/2464/66对函数接口进行注释说明/*函数功能:实现××××功能函数参数:参数1,表示××参数2,表示××函数返回值:×××××*/返回值类型函数名(形参表){…return表达式;}2022/11/2465/66挑战性的作业挑战类型表示的极限—
—50位的n!计算?–大数的存储问题2022/11/2466/66Questionsandanswers