资源预览内容
第1页 / 共105页
第2页 / 共105页
第3页 / 共105页
第4页 / 共105页
第5页 / 共105页
第6页 / 共105页
第7页 / 共105页
第8页 / 共105页
第9页 / 共105页
第10页 / 共105页
亲,该文档总共105页,到这儿已超出免费预览范围,如果喜欢就下载吧!
资源描述
第第8章章 函数和编译预处理函数和编译预处理lC语言是一种结构化程序设计语言,而结化程序设计的总体思想是采用模块化结构,自上而下,逐步求精。即首先把一个复杂的大问题分解为若干相对独立的小问题,如果小问题仍较复杂,则可以把这些小问题又继续分解成若干子问题,这样不断地分解,使得小问题或子问题简单到能够直接用程序的三种基本结构表达为止。然后,对应每一个小问题或子问题编写出一个功能上相对独立的程序块来,这种像积木一样的程序块被称为模块。掷驮郭哭脐篮君痛剔萎娠坞贴戒主泽钢彬兜娠炎寺麓出痉展獭宅气屑寝梯第8章函数和编译预处理第8章函数和编译预处理l前面各章中介绍的程序,都只有1个主函数main()。实际上,1个较大的应用程序,按结构化程序设计的要求,往往需要分成多个模块。C语言是通过函数实现模块化程序设计的,所以1个大的C语言程序,是有多个函数组成的,每个函数分别对应各自的功能模块。l在这些函数中,可以调用C编译系统提供的库函数,也可以调用自己编写的、或别人编写的自定义函数。l函数的分类:l从用户角度分为:l由系统提供标准函数(库函数)和用户自定义函数l从函数形式分为:l无参函数和有参函数矣察镇残傍鸡登佳咽俱综冻渗坏懒贡谣螟导中肥闺梭织冉炉箕食狂盯骑催第8章函数和编译预处理第8章函数和编译预处理8.1 函数的定义与调用函数的定义与调用l8.1.18.1.1函数的定义函数的定义l1任何函数(包括主函数main())都是由函数说明和函数体两部分组成。根据函数是否需要参数,可将函数分为无参函数和有参函数两种。l(1)无参函数的一般形式l 函数类型 函数名( void )l 说明语句部分;l 可执行语句部分;ll例如:voidfun(void)lprintf(“Cprogramn”);l谁藩云驹民挠烂蔫之醛莹芋窘偶代樟阶痴蓑沟挣怂铆希酒踩埂乌航挖舀烈第8章函数和编译预处理第8章函数和编译预处理l(2)有参函数的一般形式l函数类型函数类型 函数名函数名( ( 数据类型数据类型 参数参数 ,数据类型,数据类型 参参数数2 )2 )l 说明语句部分;说明语句部分;l 可执行语句部分;可执行语句部分;l l有参函数比无参函数多了一个参数表。调用有参函数时,调用函数将赋予这些参数实际的值。l为了与调用函数提供的实际参数区别开,将函数定义中的参数表称为形式参数表,简称形参表。l(3)空函数既无参数、函数体又为空的函数。其一般形式为:l 函数类型函数类型 函数名函数名(void)(void)l 员葛葫波优二郎糠叛珍友纯形寂弟抉炉沤狰稳近硼议割晕沉点膳搀标舍崭第8章函数和编译预处理第8章函数和编译预处理l 例题例题8.18.1 定义一个函数,用于求两个数中的大数。l/*功能:定义一个求较大数的函数并在主函数中调用*/lintmax(intn1,intn2)/*定义一个函数max(),n1,n2为形参*/lintz;lz=n1n2?n1:n2;/*返回n1,n1中较大者*/lreturn(z)llmain()lintmax(intn1,intn2);/*函数说明与函数定义第一行一样写,但在最后加;*/lintnum1,num2;/*num1,num2是实际参数*/lprintf(inputtwonumbers:n);lscanf(%d%d,&num1,&num2);lprintf(max=%dn,max(num1,num2);/*函数调用*/lgetch();/*使程序暂停,看结果。按任一键继续*/l运行情况:69max=9敖鸥萤钾窝况赘盲煌罗疲惕女功育陷焙性果疹肺擒间离俊冒讣恕情嚏压碎第8章函数和编译预处理第8章函数和编译预处理2说明:l(1)函数类型:指出returnreturn语句返回值的类型,它可以是C语言中任意合法的数据类型。如:int floatint float charchar等,函数类型缺省时,系统默认为int 型。l(2)函数名:是一个标识符。标识函数的名称。l(3)函数名后括号内是形式参数,写出参数的类型和名字。l如:int max(int n1, int n2)int max(int n1, int n2);不能写成:int int max(int n1, n2)max(int n1, n2);l(4)在老版本C语言中,参数类型说明允许放在函数说明部分单独指定。l例如:lint max(n1, n2)int max(n1, n2);l int n1,n2; int n1,n2;ll 玩鱼流暇呸搂哩另退术蜜伶琼滞共昨享幼砚酵覆日熊蒲仅芭褥滑籍帝挚擦第8章函数和编译预处理第8章函数和编译预处理l(5)一个C程序由一个main主函数和多个子函数组成,执行从main函数开始,调用其他函数后,返回到main函数,在main函数中结束整个程序的运行。l(6)函数定义不允许嵌套。l在语言中,所有函数(包括主函数main())都是平行的。一个函数的定义,可以放在程序中的任意位置,主函数main()之前或之后。但在一个函数的函数体内,不能再定义另一个函数,即不能嵌套定义。苏悔否抓幢私最扎菏道纷件抨衡其捏春庸菌酵整蝗待词覆恳峰订够称腔门第8章函数和编译预处理第8章函数和编译预处理8.1.2 函数的返回值与函数类型函数的返回值与函数类型l语言的函数兼有其它语言中的函数和过程两种功能,从这个角度看,又可把函数分为有返回值函数和无返回值函数两种。l1函数返回值与returnreturn语句l有参函数的返回值,是通过函数中的return语句来获得的。当然有参函数如果不需要返回值,也可以没有return语句。l(1)return语句的一般格式: return ( 返回值表达式 );或return返回值表达式;l(2)return语句的功能:返回调用函数,并将“返回值表达式”的值带给调用函数。l (3) 一个函数中可以有一个以上的return语句,执行到哪一个return语句,哪一个语句起作用。l(4)被调用函数中可以无return语句,当无return语句时并不是不返回一个值,而是一个不确定的值。为了明确表示不返回值,可以用“void”定义成“无(空)类型”。薯峰综羌催赖霍釉曙旅烬四畴哀银愚迈渭掀林迁酪晋下烙顽鞋扁淀沙捎帘第8章函数和编译预处理第8章函数和编译预处理l 例题例题8.28.2“void”定义的“无(空)类型”函数的举例lvoidf1(intx,inty)/*定义f1函数,形参有整型的x、y,void表示空类型*/lintw;lw=x+y;lprintf(“w=%dn”,w);llmain()linta,b;la=3;lb=4;lf1(a,b);/*函数的调用,实参是整型的a、b*/l皱冉客肇埋柒硒癌圆富秉氛含薯噎竣锥遮讹腔袍蘑偶育渴斗拼楚纽焙鞋炭第8章函数和编译预处理第8章函数和编译预处理2函数类型在定义函数时,对函数类型的说明,应与return语句中返回值表达式的类型一致。如果不一致,则以函数类型为准。如果缺省函数类型,则系统一律按整型处理。l 例题例题8.38.3返回值类型和函数类型不同,以函数为准。l/*例题源代码文件名:LT8_3.C*/lintmax(floatn1,floatn2)lfloatz;lz=n1n2?n1:n2;lreturn(z);/*返回n1,n1中较大者*/llmain()lfloata,b;lintc;lscanf(“%f,%f”,&a,&b);lc=max(a,b);lprintf(“maxis%dn”,c);糜艘着炸萨腿掳包灌辽汁码爹勃薪氯员毗橇坐抹邵锣拍宝悲思玖出喀斤航第8章函数和编译预处理第8章函数和编译预处理l运行情况:l1.54.5lmax=4lmax函数中return (z);z返回值是float型,而函数定义返回值类型是int型,以函数为准,所以是int型,运行结果是max=4。戮募桑旷怂泽灭继泪茬痪挖谐葬著舌欣弹抖汀虱赂疚逊趋赊我鞍泅蒙粹砧第8章函数和编译预处理第8章函数和编译预处理8.1.3 对被调用函数的说明和函数原型对被调用函数的说明和函数原型l1如果调用库函数,不用说明,但应该在本文件开头用#includeinclude命令将调用有关库函数“包含”到本文件中。l如:# include “stdio.h”# include “stdio.h” l在stdio.hstdio.h文件中放了输入输出库函数所用到的一些宏定义信息。l2如果调用自定义函数,在调用之前,应对被调用函数进行说明,其目的是:使编译系统知道被调用函数返回值的类型、函数参数的个数、类型、和顺序,便于调用时,对调用函数提供的实际参数的类型、个数、及顺序进行检查,看是否与被调用函数一致。蟹舶吮疾乍朝椅估距触积曰箭聘尉狱糕连泪徘忍题硫极十彩求骂诫唁圈鳞第8章函数和编译预处理第8章函数和编译预处理l在ANSI C新标准中,采用函数原型方式,对被调用函数进行说明,其一般格式如下:l函数类型函数类型 函数名函数名( (数据类型数据类型 参数名参数名, , 数据类型数据类型 参数名参数名2)2);l例如:例8.1主函数中的int max(int n1,int n2);语句,是函数说明语句。说明函数max的返回值类型为整型 ,有两个形式参数n1,n2都是整型。l语言同时又规定,在以下2种情况下,可以省去对被调用函数的说明:l(1) 当被调用函数的函数定义出现在调用函数之前时。因为在调用之前,编译系统已经知道了被调用函数的函数类型、参数个数、类型和顺序。l例8.1 max函数在主函数main()之前,主函数中的int max(int n1,int n2);说明语句可以不要。藩织鲍锗滇溜坏烟仕琴顾究冠羡杨姓报牧园苍钙荐凛奴帜悲骂垮蛙允季丫第8章函数和编译预处理第8章函数和编译预处理l(2) 如果在所有函数定义之前,在函数外部(例如文件开始处)预先对各个函数进行了说明,则在调用函数中可缺省对被调用函数的说明。l例如: lcharf1(inta);/*函数说明*/lfloatf2(floatb);/*函数说明*/lmain()llcharf1(inta)/*函数定义*/llfloatf2(floatb);/*函数定义*/l芋写讼叛郡筐短迂册侍陷炒穗然观滚讫敝牙捞岩问劣坪卫拒擅欧腺孽液鸿第8章函数和编译预处理第8章函数和编译预处理8.1.4 函数的调用函数的调用l在程序中,是通过对函数的调用来执行函数体的,其过程与其它语言的子程序调用相似。l1 1函数调用函数调用l语言中,函数调用的一般形式为: l 函数名函数名(实际参数表实际参数表) l(1)实参的个数、类型和顺序,应该与被调用函数所要求的参数个数、类型和顺序一致,才能正确地进行数据传递。l(2)如果实参有多个,对实际参数的求值顺序随系统而异。Turbo C按自右向左顺序求值。贤话两湾捍晒逸伟沽碰嘿撼码畏蜒钦孩甥骨阉擒臼秃井蚜艘兔某辅梯判澡第8章函数和编译预处理第8章函数和编译预处理l例 8.4 函数实际参数的求值顺序。lmain()lintf(inta,intb);/*函数说明*/linti=2,p;lp=f(i,+i);/*函数调用实参求值顺序是从右向左*/lprintf(“p=%d,p);llintf(inta,intb)/*a=3b=3*/lintc;lif(ab)c=1;lelseif(a=b)c=0;lelsec=-1;lreturn(c);l运行结果:p=0权膀绊皂抹价诡虏鳞牢薛畜氖桶滩渊挖修盒耽榴斑掺帜妊个成栽含菊茬蠢第8章函数和编译预处理第8章函数和编译预处理2函数调用方式函数调用方式l在语言中,可以用以下几种方式调用函数:l(1)函数表达式函数表达式。函数作为表达式的一项,出现在表达式中,以函数返回值参与表达式的运算。这种方式要求函数是有返回值的。l如: c=2*max(a,b); c=2*max(a,b);l(2)函数语句函数语句。C语言中的函数可以只进行某些操作而不返回函数值,这时的函数调用可作为一条独立的语句。l如:max(a,b);max(a,b);l(3)函数实参函数实参。函数作为另一个函数调用的实际参数出现。这种情况是把该函数的返回值作为实参进行传送,因此要求该函数必须是有返回值的。l 斗盂邦曾闷饼帛掩妹瞥帧谆撕慨底粥袜庙袄侩羌棱岿茅绘翅拂警棺斟狸硝第8章函数和编译预处理第8章函数和编译预处理l如:n=max(a,max(b,c);n=max(a,max(b,c);l其中max(b,c)max(b,c)是一次调用,它的值作为max另一次调用的实参。l(4)调用时实参与形参在类型必须匹配。如果类型不匹配,C编译程序将按赋值兼容的规则进行转换。(如:实参实型a=3.5,而形参x 为整型,则x得到的是3)。如果实参和形参的类型不赋值兼容,通常并不给出出错信息,且程序仍然继续执行,只是得不到正确的结果。撩鼓盛犬族揉羌甄谓移煤妙祭迫垒眷囊骸爽梁爹躺坚弃什脂耘腊忧部脸土第8章函数和编译预处理第8章函数和编译预处理8.1.5 函数的形参与实参函数的形参与实参l函数的参数分为形参形参和实参实参两种,作用是实现数据传送。形参出现在函数定义中,只能在该函数体内使用。发生函数调用时,调用函数把实参的值复制1份,传送给被调用函数的形参,从而实现调用函数向被调用函数的数据传送。l关于形参与实参的说明: l(1)实参可以是常量、变量、表达式、函数等。无论实参是何种类型的量,在进行函数调用时,它们都必须具有确定的值,以便把这些值传送给形参。因此,应预先用赋值、输入等办法,使实参获得确定的值。l(2)形参变量只有在被调用时,才分配内存单元;调用结束时,即刻释放所分配的内存单元。漓鼎摈厕易赫痞能污蔚朔蝇椽椿搏养嘎毖男枣到毫答穷陨祈裂石驭争祥氰第8章函数和编译预处理第8章函数和编译预处理l因此,形参只有在该函数内有效。调用结束,返回调用函数后,则不能再使用该形参变量。l(3)实参对形参的数据传送是单向的,即只能把实参的值传送给形参,而不能把形参的值反向地传送给实参。l(4)实参和形参占用不同的内存单元,即使同名也互不影响。如以下程序:l 例题例题8.58.5 实参对形参的数据传递。l/*功能:实参对形参的数据传递。*/l/*例题源代码文件名:LT8_5.C*/叉饰辗犀猛胶鸳呵硕吏潘宙简灶逸能锹踞柔漓泳镑渗须阔卤红岗西法禁吼第8章函数和编译预处理第8章函数和编译预处理lmain()lvoids(intn);/*说明函数*/lintn=100;/*定义实参n,并初始化*/ls(n);/*调用函数*/lprintf(n_s=%dn,n);/*输出调用后实参的值,便于进行比较*/llvoids(intn)linti;lprintf(n_x=%dn,n);/*输出改变前形参的值*/lfor(i=n-1;i=1;i-)n=n+i;/*改变形参的值*/lprintf(n_x=%dn,n);/*输出改变后形参的值*/l运行结果:n_x=100n_x=5050n_s=100肘收讨息汹椎隶衬搏梨防倦舵遭牧蛮卑怔焊仇绞巨悬跺徐暗埔受襟葵狮奶第8章函数和编译预处理第8章函数和编译预处理8.2函数的嵌套调用和递归调用函数的嵌套调用和递归调用l8.2.18.2.1函数的嵌套调用函数的嵌套调用 l函数的嵌套调用是指,在执行被调用函数时,被调用函数又调用了其它函数。这与其它语言的子程序嵌套调用的情形是类似的,其关系可表示如图8-1。郸坊疮挫闭犁咒妆遣意几恳木扰耀犯蒸溺象揉说服歹授拥刻疙吊诉想佩征第8章函数和编译预处理第8章函数和编译预处理l例题例题8.6计算s=1k+2k+3k+Nkl/*例题源代码文件名:LT8_6.C*/l/*功能:函数的嵌套调用*/l#defineK4l#defineN5llongf1(intn,intk)/*计算n的k次方*/llongpower=n;linti;lfor(i=1;ik;i+)power*=n;lreturnpower;l景楔阜库瞥转财磊省幂体爆挖箔达迭汗生匙擂锈印啸鸦喊狄菇奋粹截豆清第8章函数和编译预处理第8章函数和编译预处理llongf2(intn,intk)/*计算1到n的k次方之累加和*/llongsum=0;linti;lfor(i=1;i1)l/*例题源代码文件名:LT8_7.C*/*功能:通过函数的递归调用计算阶乘*/llongp(intn)llongf;lif(n1)f=p(n-1)*n;lelsef=1;lreturn(f);l股喝梦素猪关导核蛮娟善演堕掂症绷答欣闰眺荡拿雾卯酬疮霓署擒逛健声第8章函数和编译预处理第8章函数和编译预处理lmain()lintn;llongy;lprintf(inputainteagernumber:);lscanf(%d,&n);ly=p(n);lprintf(%d!=%ldn,n,y);lgetch();l运行结果:inputainteagernumber:55!=120熙录明奄故缮油勇箕潜郁烟氟锡叁黔浦锐讲杀昭鸳岸肢汪懒劳彬病从恶椎第8章函数和编译预处理第8章函数和编译预处理斥治虫驳铣棵饺咱吨贴焰憾层路句毁唆纵瞒卑学斟司贫售袖乖览龄滥侯学第8章函数和编译预处理第8章函数和编译预处理l 例题例题8.88.8 填空题:以下程序的输出结果是:_l/*例题源代码文件名:LT8_8.C*/lfunc(intx)lintp;lif(x=0|x=1)return(3);lp=x-func(x-2);lreturnp;llmain()llprintf(“%dn”,func(9);l噪侍赖豆耪撮弦盆捣晴又井罗规霓印跪絮色惺轮丰凛碰份工供防裂族浊汪第8章函数和编译预处理第8章函数和编译预处理分析:所以程序的输出结果是:7湿砒梯矗廉豢窟填纯岂礼萎宋卢搐拢屑傲蹿赣诺泉叙贞邪兽音庙狡苟俘耘第8章函数和编译预处理第8章函数和编译预处理8.3 数组作为函数参数数组作为函数参数l数组用作函数参数有两种形式:一种是把数组元素(又称下标变量)作为实参使用;l另一种是把数组名作为函数的形参和实参使用。 l8.3.1 8.3.1 数组元素作为函数参数数组元素作为函数参数l数组元素就是下标变量,它与普通变量并无区别。数组元素只能用作函数实参函数实参,其用法与普通变量完全相同:在发生函数调用时,把数组元素的值传送给形参,实现单向值传送。撕莫陕绦矗疾哼金裴贱快跌写颁军皆即烙金邓炉硼萍七财诵塌绑簇簧浸诞第8章函数和编译预处理第8章函数和编译预处理l 例题例题8.98.9写一函数统计字符串中字母的个数。l/*例题源代码文件名:LT8_9.C*/l/*功能:数组元素作为函数实参*/lintisalp(charc)lif(c=a&c=A&c=Z)lreturn(1);lelsereturn(0);l岁鹰轨尼篷亩航拘柿墅火衣酚犁柒刺芭峨佬栏英诸灵软戈稀奖绎匝毅胯叼第8章函数和编译预处理第8章函数和编译预处理lmain()linti,num=0;lcharstr255;lprintf(Inputastring:);lgets(str);lfor(i=0;stri!=0;i+)lif(isalp(stri)num+;/*数组元素作为实际参数调用函数*/lputs(str);/*输出字符串*/lprintf(num=%dn,num);/*输出字母个数*/l运行结果:Inputastring:WestudyTurboC!WestudyTurboC!num=13庭隔底钞郧埠鹰匀召拈垣殿睦吁输噎荧称其溯塑橱促振壬想姑哲云倦顺么第8章函数和编译预处理第8章函数和编译预处理l本例题子函数功能判断一个字符是否为字母,是返回值为1,否则为0。主函数调用时用语句for(i=0;stri!=0;i+)for(i=0;stri!=0;i+)lif (isalp(stri) num+;if (isalp(stri) num+;lisalp(stri)isalp(stri)返回值为1,表示是真,用num+num+统计字母的个数,isalp(stri)isalp(stri)返回值为0,则什么也不做。l说明:l(1)用数组元素作实参时,只要数组类型和函数的形参类型一致即可,并不要求函数的形参也是下标变量。换句话说,对数组元素的处理是按普通变量对待的。l(2)在普通变量或下标变量作函数参数时,形参变量和实参变量是由编译系统分配的两个不同的内存单元。在函数调用时发生的值传送,是把实参变量的值赋予形参变量。迄筷盅掖泽郴贝迢愧坏乍友匪茂觉窒凋景腐萤浦用惠茹侍苦院合豺妊变尧第8章函数和编译预处理第8章函数和编译预处理8.3.2 数组名作为函数的形参和实参数组名作为函数的形参和实参l数组名作函数参数时,既可以作形参,也可以作实参。l数组名作函数参数时,要求形参和相对应的实参都必须是类型相同的数组(或指向数组的指针变量),都必须有明确的数组说明。l 例题例题8.108.10已知某个学生5门课程的成绩,求平均成绩。l/*例题源代码文件名:LT8_10.C*/lfloataver(floata)/*求平均值函数*/linti;lfloatav,sum=a0;lfor(i=1;i5;i+)sum+=ai;lav=sum/5;lreturnav;l载弹颜烁闯拭芬褂验敏幕球求绰排拆舔菱乃耀萎其弱家骚使果能昂汽晾随第8章函数和编译预处理第8章函数和编译预处理lvoidmain()lfloats5,av;linti;lprintf(ninput5scores:n);lfor(i=0;i5;i+)scanf(%f,&si);lav=aver(s);/*调用函数,实参为一数组名s*/lprintf(averagescoreis%5.2fn,av);l运行结果:input5scores:8070659075averagescoreis76.00已揖窑蔼厩改珠蔓踪瑰贫酞沛峡倡癌裴肝枯恒铱注罐曰飘乓雀受需犀光胺第8章函数和编译预处理第8章函数和编译预处理l说明说明:l(1)用数组名作函数参数,应该在调用函数和被调用函数中分别定义数组,且数据类型必须一致,否则结果将出错。例如,在本案例中,形参数组为a ,实参数组为s ,它们的数据类型相同。(形参数组与实参数组可以同名)l(2)C编译系统对形参数组大小不作检查,所以形参数组可以不指定大小。例如,本案例中的形参数组a 。l为了在被调用函数中处理数组元素的需要,可以另设一个参数,传递数组元素的个数。l例如:将上例中的aver( ) 改进为如下:凯鸦泣锹坑腋光殷研嚣莆没敝僳噎卤肩全雄套瞥掸壳揩增诲鸳琐烧哈畸霉第8章函数和编译预处理第8章函数和编译预处理l例题例题8.11lfloataver(floata,intn)/*求平均值函数*/linti;lfloatav,sum=a0;lfor(i=1;in;i+)sum+=ai;lav=sum/n;lreturnav;lmain()lfloats15=98.5,97,91.5,60,55;lfloats210=98,85,75,70,60,65,77,88,90,66;lprintf(“aver1=%6.2fn”,aver(s1,5);/*调用函数,实参为一数组名s1*/lprintf(“aver2=%6.2fn”,aver(s2,10);l运行结果:aver1=80.40aver2=77.4土镜炳倒挞衬籽昂募寇脾冕蓑撬泣章绦氰媚婿技慈粱睹编料奎峰亦寞苦随第8章函数和编译预处理第8章函数和编译预处理l可以看出,两次调用aver( )函数时数组大小是不同的,在调用时用一个实参传递数组大小给形参n,以便aver( )函数能处理数组大小不同的数组元素。l如果指定形参数组的大小,则实参数组的大小必须大于等于形参数组,否则因形参数组的部分元素没有确定值而导致计算结果错误。淀示骋迸悦得万衡谱掏天孺卷朔吁嗜另庆哈联被抉旬援膀莉鹊袄拂趾央谩第8章函数和编译预处理第8章函数和编译预处理l(3)数组名作为参数,只是将实参数组的首地址传给形参数组,从而使形参数组与实参数组共用同一段内存空间。假设s1数组的起始地址为1000,则a数组的起始地址也是1000l显然形参数组和实参数组为同一数组。s0与a0同占一个单元。所以如果形参数组元素的值发生变化,也就是实参数组元素发生变化。这种数据传递方式称为“地址传递”。 蹄憎惶轮第潦凌拥栽梗秒卸戏瓤北悍狭董过以里汀献尤版炯仆啮葬触循披第8章函数和编译预处理第8章函数和编译预处理8.4 函数编程举例函数编程举例l1 1编程举例编程举例l例8.12编程求sum=1+1/2+1/3+1/4+1/5+1/nl/*例题源代码文件名:LT8_12.C*/l#include“stdio.h”lmain()ldoublefun(intn);/*函数的说明,注意末尾要加;*/lintn;lscanf(“%d”,&n);lprintf(“sum=%f”,fun(n);/*函数调用*/lldoublefun(intn)/*函数定义,末尾不能有;这是函数定义与函数说明不同的地方*/ldoublesum=0.0;境湿辱左肃冗搽链讲唤悬司逊乖翼袒艰守贞泄问若茵勋滚剔讯赌牧耗攀搀第8章函数和编译预处理第8章函数和编译预处理linti;lfor(i=1;i=n;i+)lsum+=1.0/i;lreturn(sum);ll注意:l本例计算1+1/2+1/31/n.由于“/”除号当除数和被除数均为整型时,商为整数,1/2=0、1/3=0所以要这样写sum+=1.0/i; 被除数是实型1.0除数是整型作“/”除法sum才能得到实数各项和。运行时输入10输出结果为:sum=2.928968舞推地喂旷虾漂碑总世翻蚜姚洱胃咋矢沃钙疙衔决介迭挝耕药踪云浓灵铆第8章函数和编译预处理第8章函数和编译预处理2阅读程序训练阅读程序训练l例8.13以下程序运行结果是_10,20_.lfunc(inta,intb)linttemp;ltemp=a;la=b;lb=temp;llmain()lintx,y;lx=10;y=20;lfunc(x,y);lprintf(“%d,%dn”,x,y);ll(a)10,20(b)10,10(c)20,10(d)20,20答案:(a)解析:这里是实参x,y向形参a,b值传递,在函数中只将形参a,b的值交换。不会改变实参的值,所以输出x,y的值仍为10,20。掷窘雕徽郡琶娟慢楷贱触哩群沧氖涂挎匝迷帜僧浑穗辨踊啄但西雍蓑镶冰第8章函数和编译预处理第8章函数和编译预处理l例题例题8.14以下程序运行结果是_0246_.l/*例题源代码文件名:LT8_14.C*/lvoidf1(intb)lintj;lfor(j=0;j4;j+)lbj=2*j;llmain()linta=5,6,7,8,i;lf1(a);lfor(i=0;ib?a:b;lreturn(c);llmain()linta=8;/*a为主函数的内部变量*/lprintf(“%d”,max(a,b);l运行结果:8皂心杠渐客詹闺厄霉疯瘸汗联糊铭乳栖裂果盔南哄笺剁霸帽杉惧耪摘棵派第8章函数和编译预处理第8章函数和编译预处理l本例在主函数中调用时,max(a,b)中,a是主函数的内部变量起作用,外部变量a将被屏蔽而不起作用。因此a=8 b=5进行调用,结果返回8。l3. 外部变量的作用域是从定义点到本文件结束。如果本文件定义点之前的函数需要引用这些外部变量或在其他文件中要引用时,需要用extern来声明外部变量,以扩展外部变量的作用域。l外部变量说明的一般形式为:lextern extern 数据类型数据类型 外部变量外部变量 ,外部变量,外部变量22; 爷荣据裳臃躬找宙招助律祷讳粥奔漂诛爷痴烂喳黄备息炔厄抠稚袄架塌秘第8章函数和编译预处理第8章函数和编译预处理l在一个文件内声明外部变量l如果外部变量不在文件的开头定义,其有效的作用范围只限于定义处到文件末尾。如果本文件定义点之前的函数需要引用这些外部变量,则如下例方法进行引用。l 例题例题8.178.17用extern声明外部变量,扩展程序文件中的作用域。l/*例题源代码文件名:LT8_17.C*/拴订熟逝促毙垫梗扎拆滇温疑电归它勉贴斥眷砰侮扑换臣体榆漫届付术独第8章函数和编译预处理第8章函数和编译预处理lintmax(intx,inty)lintz;lz=xy?x:y;lreturn(z);llmain()lexterna,b;/*在引用前外部变量声明*/lprintf(“%d”,max(a,b);/*引用外部变量*/llinta=13,b=-8;/*定义外部变量在后*/运行结果:13驭吧敖斩呛济渣先叼戍产绸逛冉矛弟壬已榴弘疚第缉陈办愁勾畸织方塑屯第8章函数和编译预处理第8章函数和编译预处理l(2)在多个文件的程序中声明外部变量l在一个文件中定义外部变量,而另一个文件中用extern进行声明 。l 例题例题8.188.18用extern将外部变量的作用域扩展到其他文件。l文件LT8_18a.C的内容:linta;lmain()lintb,c,m=3;lscanf(“%d%d”,&a,&b);lc=a*b;lprintf(“%dn”,c);lfun(m);lprintf(“%d”,fun(m);l掂挝垂坠球亲形虐晒秽熬狼鄂苹墅村瓦幂截硅蓄姑肉锻龋噬晶另毫降蹄熔第8章函数和编译预处理第8章函数和编译预处理l文件LT8_18b.C的内容:lexterna;/*声明本文件LT8_18b.C中的a是一个已经在其他文件中定义过的外部变量*/lfun(intn)linty;ly=a+n;lretirn(y);ll用以上方法应十分慎重,因为在执行一个文件中的函数时,可能会改变该外部变量的值,它会影响到另一个文件的函数执行结果。悟旬蚌靴把页玻咸通兽坚具皋君孩勃港辱尧裸汾鲤犯头乞蜜柄绩帽粤硼稀第8章函数和编译预处理第8章函数和编译预处理l4在定义外部变量时加static,则外部变量只限于被本文件引用,而不能被其他文件引用。l文件file1.c的内容:lstatic int a;lmain()llll则a变量只能在文件file1.c中使用。(定义是不加static就可以用extern扩展其作用域范围。如前面第3点所说)妈咨溃儒赛骡灰曼袁榔攀埠挥横超堵衅可杭杏虹唉蜜宵贴寥琅垄算遏西上第8章函数和编译预处理第8章函数和编译预处理8.6 变量的存储类别变量的存储类别l8.6.18.6.1动态存储方式和静态存储方式动态存储方式和静态存储方式l上一节从变量的作用域(即从空间)角度来分,分为:内部变量和外部变量。l从另一角度,从变量值存在的时间(即生存期)角度来分,可分为:动态存储方式和静态存储方式。l所谓静态存储方式是指在程序运行期间分配固定的存储空间的方式。而动态存储方式则是在程序运行期间根据需要进行动态的分配存储空间的方式。l内存中供用户使用的存储空间情况如图8-5l1.程序区 l2.静态存储区l3.动态存储区歇殷丹疼霍惦痢梧图桩随丈漳驼爹瘟葱浩枉堰鬼匆恨凄瘴低瘴效妒砧划寐第8章函数和编译预处理第8章函数和编译预处理l静态存储区存放外部变量。l动态存储区存放以下数据:l函数的形参。在调用函数时给形参分配存储空间。l未加static声明的内部变量(即自动变量,后面介绍)l函数调用时的现场保护和返回地址。l8.6.28.6.2自动变量自动变量l函数中的内部变量,如果定义时不加static,就是自动变量。是动态分配存储空间的。函数的形参和在函数中定义的变量,都属于此类。在调用该函数时系统会给它们分配存储空间,在函数调用结束就自动释放这些存储空间。自动变量用关键字auto作存储类别的声明。搓锁文枕戮则惹都效赊霞累瞬菇拽咯介牢周照评摆迄凌惠值念祁晶拐帝信第8章函数和编译预处理第8章函数和编译预处理l例如:lint f(int a)int f(int a)l auto int b,c=3; auto int b,c=3;ll la是形参,b b,c c是自动变量,执行完f函数后,自动释放a,b,ca,b,c所占的存储空间。关键字”auto”可以省略。不写auto auto 隐含表示是自动变量。lauto int b,c;auto int b,c; 和int b,c;int b,c;是等价的l关于自动变量的说明:l1自动变量属于动态存储方式。在函数中定义的自动变量,只在该函数内有效;函数被调用时分配存储空间,调用结束就释放。疹篷班及掸刺混辩屯奥疮葡缸傣棵励能随镭乌缄乘酮耪跋羞忍宏氦扇魄详第8章函数和编译预处理第8章函数和编译预处理l在复合语句中定义的自动变量,只在该复合语句中有效;退出复合语句后,也不能再使用,否则将引起错误。l2定义而不初始化,则其值是不确定的。如果初始化,则赋初值操作是在调用时进行的,且每次调用都要重新赋一次初值。l3由于自动变量的作用域和生存期,都局限于定义它的个体内(函数或复合语句),因此不同的个体中允许使用同名的变量而不会混淆。即使在函数内定义的自动变量,也可与该函数内部的复合语句中定义的自动变量同名。砂辙唇蒋糖崩唱为闭吁盆渐觉寐绽馋敖姓遏零颠詹柳贪涸傀捏刽氧欢妖改第8章函数和编译预处理第8章函数和编译预处理8.6.3用用static声明的内部变量声明的内部变量l有时希望函数中的内部变量的值在函数调用结束后不消失而保留原值,在下一次调用该函数时,该变量已有值,就是上一次函数调用结束的值。这时用“静态内部变量”。l1定义静态内部变量方式:lstatic 数据类型 内部变量表;l存储特点:l(1)静态内部变量属于静态存储。在程序执行过程中,即使所在函数调用结束也不释放。换句话说,在程序执行期间,静态内部变量始终存在,但其它函数是不能引用它们的,只能本函数引用。l(2)定义但不初始化,则自动赋以(整型和实型)或0(字符型);粳介坏织床腕挟壬肪称枝楷谗谈堑盈轮丫萨谣堑依德斥获毖禹澜檀堂免契第8章函数和编译预处理第8章函数和编译预处理l(3)对静态内部变量是在编译时赋初值的,即只赋初值一次,在程序运行时,它已有初值。以后每次调用它们所在的函数时,不再重新赋初值而只是保留上次函数调用结束时的值。l 例题例题8.198.19静态内部变量的使用。l/*例题源代码文件名:LT8_19.C*/lf(inta)lautointb=0;/*动态内部变量(自动变量)*/lstaticintc=3;/*静态内部变量*/lb=b+1;钾婪季押沥墟乏奏眩宙访狠兑瞧赶启庶痹诊葵辫豺班催羹磐惋跳疥畴褪产第8章函数和编译预处理第8章函数和编译预处理lc=c+1;lreturn(a+b+c);llmain()linta=2,x;lfor(x=0;x3;x+)lprintf(“%d”,f(a);l运行结果:789闽车笔稀伞躺黄赏励范吱额邯腐拓关太开酋砷敢唁椿渗第快侯啄秃著港害第8章函数和编译预处理第8章函数和编译预处理例题8.19调用过程各变量值列表第几次调用调用时初值调用结束时的值bcbca+b+c第一次03147第二次04158第三次05169维滩坝有黄笆核厅录碍甘襟谍揪浙曝宏铡扑在腕鸟杏导讫馒痊芽巷保分曝第8章函数和编译预处理第8章函数和编译预处理8.6.4寄存器变量寄存器变量registerl一般情况下,变量的值都是存储在内存中的。为提高执行效率,语言允许将局部变量的值存放到寄存器中,这种变量就称为寄存器变量。(寄存器存取速度快)l定义格式如下:l register register 数据类型数据类型 变量表;变量表;l(1)只有局部动态变量和形参才能定义成寄存器变量,即全局变量不行。l(2)对寄存器变量的实际处理,随系统而异。例如,微机上的MSC和TC 将寄存器变量实际当作自动变量处理。l(3)允许使用的寄存器数目是有限的,不能定义任意多个寄存器变量。l(4)局部静态变量不能定义为寄存器变量。焉吠拟馈酒览胀操学望观之遮稿吏怜槛芭具向撬频绦靠问帆匈径拢鹏绵任第8章函数和编译预处理第8章函数和编译预处理8.7 内部函数和外部函数内部函数和外部函数l当一个源程序由多个源文件组成时,语言根据函数能否被其它源文件中的函数调用,将函数分为内部函数内部函数和外部函数外部函数。l8.7.1 8.7.1 内部函数(又称静态函数)内部函数(又称静态函数)l如果在一个源文件中定义的函数,只能被本文件中的函数调用,而不能被同一程序其它文件中的函数调用,这种函数称为内部函数。l定义一个内部函数,只需在函数类型前再加一个“static”关键字即可,如下所示:lstatic static 函数类型函数类型 函数名函数名( (函数参数表函数参数表) )l l关键字“static”“static”,译成中文就是“静态的”,所以内部函数又称静态函数。但此处“static”的含义不是指存储方式,而是指对函数的作用域仅局限于本文件。止雹锣贡惑筷茶辽隔蛀叹姐脐湛吊旷舀固媚药发样钢垛谦寿誊限站拎枉蹦第8章函数和编译预处理第8章函数和编译预处理8.7.2 外部函数外部函数l如果在一个源文件中定义的函数,除可被本文件中的其他函数调用外,也可被其他文件中的函数所调用,这种函数称为外部函数。外部函数在整个源程序中都有效。l外部函数的定义:在定义函数时,如果没有加关键字“static”,或冠以关键字“extern”,表示此函数是外部函数:lextern extern 函数类型函数类型 函数名函数名( (函数参数表函数参数表) )l l调用外部函数时,需要对其进行说明:lextern extern 函数类型函数类型 函数名函数名( (参数类型表参数类型表),函数名,函数名2(2(参数类型表参数类型表2)2);奈拯狞震彬厢门摸确汀归返钾虐毡焦痔铃沙粉额起要对兴残筏靛笺树陵择第8章函数和编译预处理第8章函数和编译预处理如:外部函数应用l(1)文件mainf.clmain()lexternvoidinput(),process(),output();/*对外部函数进行说明*/linput();process();output();/*调用外部函数*/ll(2)文件subf1.cllexternvoidinput()/*定义外部函数*/ll(3)文件subf2.cllexternvoidprocess()/*定义外部函数*/ll(4)文件subf3.cllexternvoidoutput()/*定义外部函数*/l难蹋砧救豹亚咐敬例酵达缠笺六墓放凿剂刊圈崔陋锋劈酵奶潦歪陵潭角雨第8章函数和编译预处理第8章函数和编译预处理8.7.3 多个源程序文件的编译和连接多个源程序文件的编译和连接l1 1、用、用Turbo CTurbo C集成环境(以上面集成环境(以上面4 4个文件组成的程序为个文件组成的程序为例)例)l(1)先后分别输入并编辑文件mainf.c, 文件subf1.c, 文件subf2.c, 文件subf3.c存盘。l(2)创建Project(项目)文件:l用编辑源文件相同的方法,创建一个扩展名为.PRJ的项目文件,该文件中仅包括将被编译、连接的各源文件名,一行一个,其扩展名.C可以缺省;文件名的顺序,仅影响编译的顺序,与运行无关。l项目文件的内容为:lmainf.cmainf.clsubf1.csubf1.clsubf2.csubf2.clsubf3.csubf3.c运梦埃泄考舅军椭诽去武胀周淤撩题梧锋工志筛钞布纪碑斌叙背模哪炊孝第8章函数和编译预处理第8章函数和编译预处理l(3)设置项目名称:l打开菜单,选取Project菜单选Project name项按回车,屏幕出现一个对话框,询问项目文件名,输入项目文件名a.prj, 其作用是表示当前准备编译的是a.prj中包括的文件。l(4)按功能键F9进行编译、连接,系统先后将4个文件翻译成目标文件,并把它们连接成一个可执行文件a.exe。l(5)按ctrl+F9运行可执行文件a.exe。l与单个源文件相同。编译产生的目标文件,以及连接产生的可执行文件,它们的主文件名,均与项目文件的主文件名相同。l注意注意:当前项目文件调试完毕后,应选取ProjectClear project,将其项目名称从“Project name”中清除(清除后为空)。否则,编译、连接和运行的,始终是该项目文件!青题晓疆眉佬伸类忌平硷遇牲勾呢啡困婆芹衰絮垦烈柒绅测室随想岔维肾第8章函数和编译预处理第8章函数和编译预处理8.8编译预处理编译预处理l所谓编译预处理是指在对源程序进行编译之前,先对源程序中的编译预处理命令进行处理;然后再将处理的结果,和源程序一起进行编译,以得到目标代码。 l编译预处理是C语言的一个重要功能,由编译预处理程序完成。为了与 C语言语句区别开,编译预处理命令以#号打头,单独占用一个书写行,行尾不使用分号作为结束符。除持埂斥托萌履卢烙沁畔王匡入乞烩和捞笼论温瞻焙磺进慷劈矩共非炕掉第8章函数和编译预处理第8章函数和编译预处理8.8.1 宏定义与宏展开宏定义与宏展开l在语言中,“宏”分为无参数的宏(简称无参宏)和有参数的宏(简称有参宏)两种。l1 1无参宏定义无参宏定义 l(1)无参宏定义的一般格式l #define #define 标识符标识符 语言符号字符串语言符号字符串l其中:“define”为宏定义命令;“标识符”为所定义的宏名,通常用大写字母表示,以便于与变量区别;“语言符号字符串”可以是常数、表达式等。鸽洱帧松阐硬谱要糯萝楼都愁阜夕肝抚崭唇量果吨吵鞭僵页邢辣神迎鲜伞第8章函数和编译预处理第8章函数和编译预处理例题例题8.20输入圆的半径,求圆的周长、面积和球的体积。要求使用无参宏定义圆周率。l#definePI3.1415926/*PI是宏名,3.1415926用来替换宏名的常数*/lmain()lfloatr,len,s,v;lprintf(Inputaradius:);lscanf(%f,&r);llen=2*PI*r;/*引用无参宏求周长*/ls=PI*r*r;/*引用无参宏求面积*/lv=PI*r*r*r*3/4;/*引用无参宏求体积*/lprintf(len=%.2f,s=%.2f,v=%.2fn,len,s,v);l编译预处理后,宏展开的结果如下:len=2*3.1415926*r;s=3.1415926*r*r;/*用3.1415926替换宏名PI*/v=3.1415926*r*r*r*3/4;颓治哨锥萎肯迂个现遭罚输砷裙谰澎集朽拐卞蹿吝戴护升降批袖头羊揍咽第8章函数和编译预处理第8章函数和编译预处理(2)说明l宏名一般用大写字母表示,以示与变量区别。但这并非是规定。 l 宏定义不是语句,所以不能在行尾加分号。否则,宏展开时,会将分号作为字符串的1个字符,用于替换宏名。l 宏展开时,预处理程序仅以按宏定义简单替换宏名,而不作任何检查。如果有错误,只能由编译程序在编译宏展开后的源程序时发现。l如:# define PI 3.I4I59l把数字1写成字母I,预处理时也照样代入,不管含义是否正确,不作任何检查。l宏定义命令#define出现在函数的外部,宏名的有效范围是:从定义命令之后, 到本文件结束。通常,宏定义命令放在文件开头处。l如果需要,也可以#undef命令终止宏名的作用域。如:蓉峰斡硕挪亲渠蠕米仁挚好姜正贪湍晓惭驮韩棘肆径垫痊壳磁禄检窟柔哪第8章函数和编译预处理第8章函数和编译预处理l#define PI 3.1415926#define PI 3.1415926lmain( )main( )ll#undef PI#undef PI /*终止宏名PI的作用域,即PI只在main( )函数中有效*/lvoid fun()void fun()ll 在进行宏定义时,可以引用已定义的宏名 。l#define PI 3.1415926#define PI 3.1415926l#define R 2.5#define R 2.5l#define LEN 2*PI*R#define LEN 2*PI*R /*引用已定义的宏名PI 和R*/l恋腺摘著夹叼慎艾撞举峡浩碾幅顿泌冉援烈醒端篡痕善虾巳荐乾制伯裕毋第8章函数和编译预处理第8章函数和编译预处理l对双引号括起来的字符串内的字符,即使与宏名同名,也不进行宏展开。 l#definePI3.14lmain()lprintf(“PI=%.2fn”,PI);ll运行结果:lPI=3.14l使用宏名代替一个字符串,可以减少程序中重复书写某些字符串的工作量。如:如果不定义PI代表3.1415926,则在程序中要多处出现3.1415926,不仅麻烦,而且容易写错,用宏名代替,简单不易出错。另外当需要改变某一个常量时,可以只改变#define命令行,一改全改。把沽绑怂筐晾杠吊鹅鄂渊汇涕罚逻拳伺孺炕决欠蓖赂钥罕转因削秆筛踢奏第8章函数和编译预处理第8章函数和编译预处理l宏定义是专门用于预处理命令的一个专用名词,它与定义变量的含义不同,只作字符替换,不分配内存空间。l2 2有参宏定义有参宏定义 l带参宏定义的一般格式l#define #define 宏名宏名( (形参表形参表) ) 语言符号字符串语言符号字符串l带参宏的调用和宏展开l(1)调用格式:宏名宏名( (实参表实参表) )l(2)宏展开:用宏调用提供的实参字符串,直接置换宏定义命令行中、相应形参字符串,非形参字符保持不变。l#define S(a,b) a*b#define S(a,b) a*blarea=S(3,2);area=S(3,2);是宏展开,展开后为:area=3*2;area=3*2;咕曙捌捏粮监差固茵位诵红氢捏尸个咕汞迁捡诗僳片移槐狂凿辨拆估盂附第8章函数和编译预处理第8章函数和编译预处理l说明:l 定义有参宏时,宏名与左圆括号之间不能留有空格。否则,编译系统将空格以后的所有字符均作为替代字符串,而将该宏视为无参宏。l 有参宏的展开,只是将实参作为字符串,简单地置换形参字符串,而不做任何语法检查。在定义有参宏时,在所有形参外和整个字符串外,均加一对圆括号。 l 例题例题8.218.21l/*例题源代码文件名:LT8_21.C*/l#definePI3.14l#defineS(r)PI*r*r鼻伶签译捧戎演禾蝎任吐把晃崇腰抓用钱谁胁钾件刺郧情拈版凄厉陋撰种第8章函数和编译预处理第8章函数和编译预处理lmain()lfloata,b,area;la=3.5;lb=2.5;larea=S(a+b);lprintf(“area=%fn”,area);ll运行结果:larea=22.240000l宏展开后为:area=PI*a+b*a+b;照怂搏厢蝗矾病理漆鼎麓悲绪烷抒凡涉苍咸疚组纶魄扰妮啪拄村乓森劲柏第8章函数和编译预处理第8章函数和编译预处理l请注意请注意: :在a+ba+b外面没有括号,显然这与程序设计者的原意不符合。原意希望得到larea=PI*(a+b)*(a+b);area=PI*(a+b)*(a+b);l为了得到这个结果,应该在定义时,在字符串中的形式参数外面加一个括号。即l#definr S(R) PI*(r)* (r)#definr S(R) PI*(r)* (r) l 虽然有参宏与有参函数确实有相似之处,但不同之处更多,主要有以下几个方面:l调用有参函数时,是先求出实参的值,然后再复制一份给形参。而展开有参宏时,只是将实参简单地置换形参。l在有参函数中,形参是有类型的,所以要求实参的类型与其一致;而在有参宏中,形参是没有类型信息的,因此用于置换的实参,什么类型都可以。有时,可利用有参宏的这一特性,实现通用函数功能。哈木振鲍洁草腿绕剂氦敷哥闪案卸待畜空咏罗乔谆刑撰崭眩锡懒伍素央吉第8章函数和编译预处理第8章函数和编译预处理l使用有参函数,无论调用多少次,都不会使目标程序变长,但每次调用都要占用系统时间进行调用现场保护和现场恢复;而使用有参宏,由于宏展开是在编译时进行的,所以不占运行时间,但是引用时,会使目标程序增大。 l3 3阅读程序练习:阅读程序练习: l例8.22以下程序的运行结果是_15_l/*例题源代码文件名:LT8_22.C*/l#defineMIN(x,y)(x)(y)?(x):(y)lmain()linta,b,c;la=10;譬佯堡谩于灵峪园觉颊卡诡陇檄锐徐呆玩僧臭晤诞泡吗岩锅络涛尧俊姑浚第8章函数和编译预处理第8章函数和编译预处理lb=15;lc=10*MIN(a,b);lprintf(“%d”,c);ll解析:因为有参宏展开,只是将实参作为字符串,简单地置换形参字符串,展开后为:lc=10*(10)(15)?(10):(15);然后编译、连接、执行,由于*乘法运算优先级高于条件运算,先计算10*10=100,c=100(15)?(10):(15),结果c=15熊省覆糯猩兄诈活曲花润铰贩于唯膳联谨兹赊驻搐债砰驼兰簿厉唤邀槛羚第8章函数和编译预处理第8章函数和编译预处理l例8.23l#defineN2l#defineMN+2l#defineCUBE(x)(x*x*x)lmain()linty,w;ly=M;/*展开为y=N+2*/lw=CUBE(y);/*展开为y=y*y*y*/lprintf(“%dn”,w);ll解析:y=M;展开为y=N+2, w=CUBE(y) 展开为w=y*y*y;(宏展开只是实参简单置换形参,不具有计算功能)然后编译连接执行时,y=4;w=4*4*4=64。锨碉温旬架匙渭坤霜哟柿棘饲揉柔涧敲河般澳毯兹诡丁与捍倍呈场道桶袍第8章函数和编译预处理第8章函数和编译预处理8.8.2文件包含文件包含l1 1文件包含的概念文件包含的概念l文件包含是指,一个源文件可以将另一个源文件的全部内容包含进来。虎着彬饼承门倔域痕项喇泵混痢凄烧兔兜宴郑魏从外车春超夸腑鬃捣溃狼第8章函数和编译预处理第8章函数和编译预处理2文件包含处理命令的格式文件包含处理命令的格式linclude “include “包含文件名包含文件名” 或 include include l两种格式的区别仅在于:l(1)使用双引号:系统首先到当前目录下查找被包含文件,如果没找到,再到系统指定的“包含文件目录”(由用户在配置环境时设置)去查找。l(2)使用尖括号:直接到系统指定的“包含文件目录”去查找。一般地说,使用双引号比较保险。 淫绳提靶淑帛蔽勺潭蛹切罚叮付儒孺毅孰勃纳褐萝师葱衷某淡新乌厌码峪第8章函数和编译预处理第8章函数和编译预处理3文件包含的优点文件包含的优点l一个大程序,通常分为多个模块,并由多个程序员分别编程。有了文件包含处理功能,就可以将多个模块共用的数据(如符号常量和数据结构)或函数,集中到一个单独的文件中。这样,凡是要使用其中数据或调用其中函数的程序员,只要使用文件包含处理功能,将所需文件包含进来即可,不必再重复定义它们,从而减少重复劳动。l4说明l(1)编译预处理时,预处理程序将查找指定的被包含文件,并将其复制到#include命令出现的位置上。l被包含文件应该是源文件,而不是目标文件。当被包含文件的内容发生变化时 ,包括这个文件的所有源文件都应重新编译、连接。稻豢叙内烫保痹精凿酣盖蠕泳彼侯炙擞美侮爽宾眠畴即逛捂澎湾套料效锦第8章函数和编译预处理第8章函数和编译预处理l如:l#include“stdio.h”lmain()lcharch;lch=gechar();lputchar(ch);lputchar(n);ll(2)如果程序中只用scanf()、printf()函数库函数,可以省略#include文件包含处理命令。(因为它们使用的非常多,C语言规定可以省略#include命令)秘叫菠经杯奔争瑟钻刚卜泽需杀境贱谱废马洞拒糖紊祥湛凝译迅仕私辣巷第8章函数和编译预处理第8章函数和编译预处理l(3)一条包含命令,只能指定一个被包含文件。如果要包含n个文件,则要用n条包含命令。l(4)文件包含可以嵌套,即被包含文件中又包含另一个文件。l(5)常用在文件头部分的被包含文件,称为“标题文件”或“头部文件”,常以“h”(head)作为后缀,简称头文件。在头文件中,除可包含宏定义外,还可包含外部变量定义、结构类型定义等。些吐萎刹省啡判茨烛梗丝抽网小瘩铣殷青躺半丝捡热贪娇接甄滴做蕊驾岔第8章函数和编译预处理第8章函数和编译预处理8.8.3 条件编译条件编译l一般情况下,源程序中所有的行都参加编译。但是有时,希望对部分源程序行只在满足一定条件时才编译,即对这部分源程序行指定编译条件。l条件编译可有效地提高程序的可移植性,并广泛地应用在商业软件中,为一个程序提供各种不同的版本。l1 #ifdef #endif命令命令l一般格式lifdef标识符l程序段1;lelsel程序段2;lendif羹戚运擞沂矫胺房皋孰殴邯瞳淑邪盲盒少穴呼申宋狸描辖蛇趾机屯造哮外第8章函数和编译预处理第8章函数和编译预处理l功能:功能:当“标识符”已经被#define命令定义过,则编译程序段1,否则编译程序段2。l(1)在不同的系统中,一个int 型数据占用的内存字节数可能是不同的。 l例如:IBM-PC机为2个字节,而其他计算机系统用4个字节。为了提高程序的通用性,可利用条件编译预处理功能:l#defineIBM-PCll#ifdefIBM-PCl#defineINTEGER_SIZE16l#elsel#defineINTEGER_SIZE32l#endifl志葱侵铆侩舵情塌撒中彭猖糕颐短淬昌末她反邢翰丛钧消夫侄动公霍太著第8章函数和编译预处理第8章函数和编译预处理l(2)利用条件编译,还可使同一源程序即适合于调试(进行程序跟踪、打印较多的状态或错误信息),又适合高效执行要求。 l例如:可在调试时,在程序的开头设置一个代表调试状态的符号常量DEBUG,并在需要的位置上插入以下的条件编译段:l# ifdef DEBUG# ifdef DEBUGlprintf(“x=%d,y=%d,z=%dn”,x,y,z);printf(“x=%d,y=%d,z=%dn”,x,y,z);l#endif#endifl如果在它的前面有以下命令行:l# define DEBUG# define DEBUGl则在程序运行时输出x,y,z的值,以便调试时分析。调试完成后,只要删除程序开头的# define DEBUG命令行,然后重新编译、连接。耸素嘿惠丸摔鹏疫荐浅环皖衙艺嘶你帕泥谊滩艘荡奴让管旗拈墒笋熙弯炙第8章函数和编译预处理第8章函数和编译预处理l2 2#ifndef #endif#ifndef #endif命令命令l格式与#ifdef #endif#ifdef #endif命令一样,功能正好与之相反。即当“标识符”未被#define#define命令定义过,则编译程序段1,否则编译程序段2。l3 3#if #endif#if #endifl一般格式l ifif 常量表达式l 程序段1;l elseelsel 程序段2;l endifendifl功能:当表达式为非0(“逻辑真”)时,编译程序段1,否则编译程序段2。 糊拇坎嫌钙溺从诲挝鸡题疯甩住膘聋凄终乔抓绚乾乎蒋疲倒旷驼乡木鸽钠第8章函数和编译预处理第8章函数和编译预处理l 例题例题8.248.24输入10个整型数,根据设置的编译条件,求最大值或最小值。 l#include“stdio.h”l#defineMFLAG1 /*预置为最大值*/lmain()lintx,m;linta10;lfor(x=0;x10;x+)lscanf(“%d”,&ax);lm=a0;lfor(x=1;xm)lm=ax;l#elselif(axm) m=ax;if(axm) m=ax;求最大值。l(2)当定义MFLAG为0时,for语句内循环体,编译if(axm) m=ax;if(axm) m=ax;求最小值。l有的读者会想,不用条件编译命令而直接用if语句也能达到要求,用条件编译命令有什么好处呢?的确,此问题可以不用条件编译处理,但那样做目标程序长(因为所有语句都编译),运行时间长(因为在程序运行时 对if语句进行测试)。而采用条件编译,可以减少被编译的语句,从而减少目标程序的长度,减少运行时间。拖乓屯着繁评均韶辈穗调炸甸彪部借欢棒筋肖娠涡躺且麦悯忿晤剖葵敛今第8章函数和编译预处理第8章函数和编译预处理8.9阅读程序训练阅读程序训练l 例题例题8.258.25以下程序运行结果是_lintx=1;/*x是外部变量*/lmain()lintfunc(intx);/*声明函数*/lfunc(x);/*调用函数func()*/lprintf(“x=%dn”,x);llintfunc(intx)lx=3;运行结果:x=1央杯甄勾哭都渡俱沸汲宜禽卤锦琉鞘悯琐克卿珊囊锰尸侧嚣爷澳盖疏缆斡第8章函数和编译预处理第8章函数和编译预处理解析:解析:l在函数之外的语句int x=1;中的x是外部变量,它从定义点开始直到文件末尾是一直有效的,因此主函数中调用语句func(x);中的实参x是1,在调用时传递给func函数的形参x使其值为1,在func函数中,执行x=3;语句时,因为形参x是内部变量,在函数的内部,内部变量x是屏蔽外部变量x的,所以形参x有效且值为3。当调用结束时,形参变量x=3释放无效(即消失),故返回主函数时,执行主函数的printf(“x=%dn”,x);语句,输出的x值仍是外部变量x的值1。蹭泄翁藉再谱咋怠牡需厕评咏它项缓谬玛是沮饯焙侨怜狠捍拧虑堤咎瘟炸第8章函数和编译预处理第8章函数和编译预处理解析解析:本题是将字符叔祖的内容反序存放后输出。基本算法思路是:将数组的第一个元素与最后一个元素交换位置,将数组的第二个元素与数组的倒数第二个元素交换位置,当扫描到数组中部时,交换结束。l 例题例题8.268.26以下程序运行结果是_linverse(charb)lintx,y,len=strlen(b);lchart;lfor(x=0;xlen/2;x+)lt=bx;lbx=blen-x-1;lblen-x-1=t;llmain()chara10;inverse(a);printf(“theresultis:%sn”,a);运行情况:abcdefggfedcba技锄敲京业溃冉翁矣坍譬拖摩沙滨瑶代只叫契话骄巧橡条榷呻牡曝烈童柄第8章函数和编译预处理第8章函数和编译预处理
网站客服QQ:2055934822
金锄头文库版权所有
经营许可证:蜀ICP备13022795号 | 川公网安备 51140202000112号