sq=7 同樣輸入3,但結(jié)果卻是不一樣的。問題在哪里呢? 這是由于代換只作符號(hào)代換而不作其它處理而造成的。 宏代換后將得到以下語句: sq=a+1*a+1; 由于a為3故sq的值為7。這顯然與題意相違,因此參數(shù)兩邊的括號(hào)是不能少的。即使在參數(shù)兩邊加括號(hào)還是不夠的,請(qǐng)看下面程序:
#define SQ(y) (y)*(y)
main(){
int a,sq;
printf('input a number: ');
scanf('%d',&a);
sq=160/SQ(a+1);
printf('sq=%d\n',sq);
}
本程序與前例相比,只把宏調(diào)用語句改為: sq=160/SQ(a+1); 運(yùn)行本程序如輸入值仍為3時(shí),希望結(jié)果為10。但實(shí)際運(yùn)行的結(jié)果如下:input a number:3 sq=160為什么會(huì)得這樣的結(jié)果呢?分析宏調(diào)用語句,在宏代換之后變?yōu)椋?sq=160/(a+1)*(a+1);a為3時(shí),由于“/”和“*”運(yùn)算符優(yōu)先級(jí)和結(jié)合性相同, 則先作160/(3+1)得40,再作40*(3+1)最后得160。為了得到正確答案應(yīng)在宏定義中的整個(gè)字符串外加括號(hào), 程序修改如下
#define SQ(y) ((y)*(y))
main(){
int a,sq;
printf('input a number: ');
scanf('%d',&a);
sq=160/SQ(a+1);
printf('sq=%d\n',sq);
}
以上討論說明,對(duì)于宏定義不僅應(yīng)在參數(shù)兩側(cè)加括號(hào), 也應(yīng)在整個(gè)字符串外加括號(hào)。
5. 帶參的宏和帶參函數(shù)很相似,但有本質(zhì)上的不同,除上面已談到的各點(diǎn)外,把同一表達(dá)式用函數(shù)處理與用宏處理兩者的結(jié)果有可能是不同的。
main(){
int i=1;
while(i<=5)>=5)>
printf('%d\n',SQ(i++));
}
SQ(int y)
{
return((y)*(y));
}#define SQ(y) ((y)*(y))
main(){
int i=1;
while(i<=5)>=5)>
printf('%d\n',SQ(i++));
}
在上例中函數(shù)名為SQ,形參為Y,函數(shù)體表達(dá)式為((y)*(y))。在例9.6中宏名為SQ,形參也為y,字符串表達(dá)式為(y)*(y))。 兩例是相同的。例9.6的函數(shù)調(diào)用為SQ(i++),例9.7的宏調(diào)用為SQ(i++),實(shí)參也是相同的。從輸出結(jié)果來看,卻大不相同。分析如下:在例9.6中,函數(shù)調(diào)用是把實(shí)參i值傳給形參y后自增1。 然后輸出函數(shù)值。因而要循環(huán)5次。輸出1~5的平方值。而在例9.7中宏調(diào)用時(shí),只作代換。SQ(i++)被代換為((i++)*(i++))。在第一次循環(huán)時(shí),由于i等于1,其計(jì)算過程為:表達(dá)式中前一個(gè)i初值為1,然后i自增1變?yōu)?,因此表達(dá)式中第2個(gè)i初值為2,兩相乘的結(jié)果也為2,然后i值再自增1,得3。在第二次循環(huán)時(shí),i值已有初值為3,因此表達(dá)式中前一個(gè)i為3,后一個(gè)i為4, 乘積為12,然后i再自增1變?yōu)?。進(jìn)入第三次循環(huán),由于i 值已為5,所以這將是最后一次循環(huán)。計(jì)算表達(dá)式的值為5*6等于30。i值再自增1變?yōu)?,不再滿足循環(huán)條件,停止循環(huán)。從以上分析可以看出函數(shù)調(diào)用和宏調(diào)用二者在形式上相似, 在本質(zhì)上是完全不同的。
6. 宏定義也可用來定義多個(gè)語句,在宏調(diào)用時(shí),把這些語句又代換到源程序內(nèi)。看下面的例子。
#define SSSV(s1,s2,s3,v) s1=l*w;s2=l*h;s3=w*h;v=w*l*h;
main(){
int l=3,w=4,h=5,sa,sb,sc,vv;
SSSV(sa,sb,sc,vv);
printf('sa=%d\nsb=%d\nsc=%d\nvv=%d\n',sa,sb,sc,vv);
}
程序第一行為宏定義,用宏名SSSV表示4個(gè)賦值語句,4 個(gè)形參分別為4個(gè)賦值符左部的變量。在宏調(diào)用時(shí),把4 個(gè)語句展開并用實(shí)參代替形參。使計(jì)算結(jié)果送入實(shí)參之中。
文件包含
文件包含是C預(yù)處理程序的另一個(gè)重要功能。文件包含命令行的一般形式為: #include'文件名' 在前面我們已多次用此命令包含過庫函數(shù)的頭文件。例如:
#include'stdio.h'
#include'math.h'
文件包含命令的功能是把指定的文件插入該命令行位置取代該命令行, 從而把指定的文件和當(dāng)前的源程序文件連成一個(gè)源文件。在程序設(shè)計(jì)中,文件包含是很有用的。 一個(gè)大的程序可以分為多個(gè)模塊,由多個(gè)程序員分別編程。 有些公用的符號(hào)常量或宏定義等可單獨(dú)組成一個(gè)文件, 在其它文件的開頭用包含命令包含該文件即可使用。這樣,可避免在每個(gè)文件開頭都去書寫那些公用量, 從而節(jié)省時(shí)間,并減少出錯(cuò)。
對(duì)文件包含命令還要說明以下幾點(diǎn):
1. 包含命令中的文件名可以用雙引號(hào)括起來,也可以用尖括號(hào)括起來。例如以下寫法都是允許的: #include'stdio.h' #include 但是這兩種形式是有區(qū)別的:使用尖括號(hào)表示在包含文件目錄中去查找(包含目錄是由用戶在設(shè)置環(huán)境時(shí)設(shè)置的), 而不在源文件目錄去查找; 使用雙引號(hào)則表示首先在當(dāng)前的源文件目錄中查找,若未找到才到包含目錄中去查找。 用戶編程時(shí)可根據(jù)自己文件所在的目錄來選擇某一種命令形式。
2. 一個(gè)include命令只能指定一個(gè)被包含文件, 若有多個(gè)文件要包含,則需用多個(gè)include命令。3. 文件包含允許嵌套,即在一個(gè)被包含的文件中又可以包含另一個(gè)文件。
條件編譯
預(yù)處理程序提供了條件編譯的功能。 可以按不同的條件去編譯不同的程序部分,因而產(chǎn)生不同的目標(biāo)代碼文件。 這對(duì)于程序的移植和調(diào)試是很有用的。 條件編譯有三種形式,下面分別介紹:
1. 第一種形式:
#ifdef 標(biāo)識(shí)符
程序段1
#else
程序段2
#endif
它的功能是,如果標(biāo)識(shí)符已被 #define命令定義過則對(duì)程序段1進(jìn)行編譯;否則對(duì)程序段2進(jìn)行編譯。如果沒有程序段2(它為空),本格式中的#else可以沒有, 即可以寫為:
#ifdef 標(biāo)識(shí)符
程序段 #endif
#define NUM ok
main(){
struct stu
{
int num;
char *name;
char sex;
float score;
} *ps;
ps=(struct stu*)malloc(sizeof(struct stu));
ps->num=102;
ps->name='Zhang ping';
ps->sex='M';
ps->score=62.5;
#ifdef NUM
printf('Number=%d\nScore=%f\n',ps->num,ps->score);
#else
printf('Name=%s\nSex=%c\n',ps->name,ps->sex);
#endif
free(ps);
}
由于在程序的第16行插入了條件編譯預(yù)處理命令, 因此要根據(jù)NUM是否被定義過來決定編譯那一個(gè)printf語句。而在程序的第一行已對(duì)NUM作過宏定義,因此應(yīng)對(duì)第一個(gè)printf語句作編譯故運(yùn)行結(jié)果是輸出了學(xué)號(hào)和成績。在程序的第一行宏定義中,定義NUM表示字符串OK,其實(shí)也可以為任何字符串,甚至不給出任何字符串,寫為: #define NUM 也具有同樣的意義。 只有取消程序的第一行才會(huì)去編譯第二個(gè)printf語句。讀者可上機(jī)試作。
2. 第二種形式:
#ifndef 標(biāo)識(shí)符
程序段1
#else
程序段2
#endif
與第一種形式的區(qū)別是將“ifdef”改為“ifndef”。它的功能是,如果標(biāo)識(shí)符未被#define命令定義過則對(duì)程序段1進(jìn)行編譯, 否則對(duì)程序段2進(jìn)行編譯。這與第一種形式的功能正相反。
3. 第三種形式:
#if 常量表達(dá)式
程序段1
#else
程序段2
#endif
它的功能是,如常量表達(dá)式的值為真(非0),則對(duì)程序段1 進(jìn)行編譯,否則對(duì)程序段2進(jìn)行編譯。因此可以使程序在不同條件下,完成不同的功能
#define R 1
main(){
float c,r,s;
printf ('input a number: ');
scanf('%f',&c);
#if R
r=3.14159*c*c;
printf('area of round is: %f\n',r);
#else
s=c*c;
printf('area of square is: %f\n',s);
#endif
}
本例中采用了第三種形式的條件編譯。在程序第一行宏定義中,定義R為1,因此在條件編譯時(shí),常量表達(dá)式的值為真, 故計(jì)算并輸出圓面積。上面介紹的條件編譯當(dāng)然也可以用條件語句來實(shí)現(xiàn)。 但是用條件語句將會(huì)對(duì)整個(gè)源程序進(jìn)行編譯,生成的目標(biāo)代碼程序很長,而采用條件編譯,則根據(jù)條件只編譯其中的程序段1或程序段2, 生成的目標(biāo)程序較短。如果條件選擇的程序段很長, 采用條件編譯的方法是十分必要的。
本站僅提供存儲(chǔ)服務(wù),所有內(nèi)容均由用戶發(fā)布,如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)
點(diǎn)擊舉報(bào)。