|
條件編譯就是按條件對(duì)C程序的一部分進(jìn)行編譯,其它部分不編譯。條件編譯的目的是使源代碼能更迅速、更容易地進(jìn)行修改,并使目標(biāo)代碼縮短。這樣,當(dāng)程序在不同系統(tǒng)上編譯、在同一系統(tǒng)不同編譯器上編譯或進(jìn)行不同目的的編譯時(shí),減少對(duì)程序語(yǔ)句的修改,而讓編譯預(yù)處理器把該語(yǔ)句留下或忽略。 條件編譯有#if語(yǔ)句、#if-elif語(yǔ)句、#ifdef語(yǔ)句、#ifndef語(yǔ)句。 條件編譯語(yǔ)句
1.# if # if的一般形式是: # if 表達(dá)式 程序段1 # else 程序段2 # endif # if的執(zhí)行過(guò)程是:如果表達(dá)式為真,編譯程序段1,否則編譯程序段2。作為一種特例,當(dāng)條件為假不執(zhí)行任何操作時(shí),可以省略#else。 2.# if-#elif # if-#elif的形式與if-else if語(yǔ)句形式基本相同,格式為: # if 表達(dá)式1 程序段1 … # elif 表達(dá)式m 程序段m # else 程序段n # endif 如果表達(dá)式1的值為真,編譯程序段1,否則如果表達(dá)式2為真,編譯程序段2,依此類(lèi)推,若表達(dá)式m為真,編譯程序段m,否則編譯程序段n。 3.# ifdef # ifdef的一般形式是: # ifdef 標(biāo)識(shí)符 程序段1 # else 程序段2 # endif 如果標(biāo)識(shí)符在此之前已經(jīng)由# define給出了定義,就編譯程序段1,如果這部分內(nèi)容沒(méi)有可以空著。否則編譯程序段2,如果沒(méi)有程序段2,#else也可省略。 4.# ifndef # ifndef的一般形式是: # ifndef 標(biāo)識(shí)符 程序段1 # else 程序段2 # endif 如果標(biāo)識(shí)符在此之前未經(jīng)定義,就編譯程序段1,如果這部分內(nèi)容沒(méi)有可以空著。否則編譯程序段2,如果沒(méi)有程序段2,#else也可省略。。 條件編譯例題
例1 #define MAX 10 main( ) {# if MAX>99 printf("Compiled for array greater than 99\n"); # else printf("Compiled for small array\n"); # endif } 這里MAX=10,表達(dá)式MAX>99為假,因此不編譯# if之后的程序段,而編譯#else后的程序段,結(jié)果顯示Compiled for small array。 例2 用ACTIVE_COUNTRY的值來(lái)決定貨幣符號(hào): #define US 0 #define ENGLAND 1 #define FRANCE 2 # define ACTIVE_COUNTRY US # if ACTIVE_COUNTRY==US char currency[ ]= "dollar"; # elif ACTIVE_COUNTRY==ENGLAND char currency[ ]= "pound"; # elif ACTIVE_COUNTRY==FRANCE char currency[ ]= "franc"; # endif 例3 #define ZHAO 10 main( ) { #ifdef ZHAO printf("Hello,Zhao\n"); #else printf("Hello,anyone\n"); #endif #ifndef CHEN printf("Chen not be defined\n"); #endif } 程序運(yùn)行結(jié)果: Hello,Zhao Chen not be defined 如果把本定義中的宏定義去掉,則顯示Hello, anyone和Chen not be defined。 例4 采用條件編譯,使給定的字符串按小寫(xiě)字母或大寫(xiě)字母輸出。 #include "stdio" #define LETTER 1 main( ) { int i=0; static char str[ ]={ "Turbo C Program"}; char c; while ((c=str[i] != '\0') { i++; # if LETTER if (c>='a' && c<='z') c=c-32; # else if (c>='A' && c<='Z') c=c+32; # endif printf("%c",c); } } 因?yàn)長(zhǎng)ETTER定義為1,即條件表達(dá)式的值為真,則在預(yù)處理時(shí),對(duì)第一個(gè)if語(yǔ)句進(jìn)行編譯處理,故此程序可以把小寫(xiě)字母變成大寫(xiě)字母。 程序運(yùn)行結(jié)果為: TURBO C PROGRAM 如果將程序第二行改為:#define LETTER 0則在預(yù)處理時(shí),對(duì)第二個(gè)if語(yǔ)句進(jìn)行編譯處理,把大寫(xiě)字母變成小寫(xiě)字母。 ============================================================ 這幾個(gè)宏是為了進(jìn)行條件編譯。一般情況下,源程序中所有的行都參加編譯。但是有時(shí)希望對(duì)其中一部分內(nèi)容只在滿(mǎn)足一定條件才進(jìn)行編譯,也就是對(duì)一部分內(nèi)容指定編譯的條件,這就是“條件編譯”。有時(shí),希望當(dāng)滿(mǎn)足某條件時(shí)對(duì)一組語(yǔ)句進(jìn)行編譯,而當(dāng)條件不滿(mǎn)足時(shí)則編譯另一組語(yǔ)句。 條件編譯命令最常見(jiàn)的形式為: #ifdef 標(biāo)識(shí)符 程序段1 #else 程序段2 #endif 它的作用是:當(dāng)標(biāo)識(shí)符已經(jīng)被定義過(guò)(一般是用#define命令定義),則對(duì)程序段1進(jìn)行編譯,否則編譯程序段2。 其中#else部分也可以沒(méi)有,即: #ifdef 程序段1 #denif 這里的“程序段”可以是語(yǔ)句組,也可以是命令行。這種條件編譯可以提高C源程序的通用性。如果一個(gè)C源程序在不同計(jì)算機(jī)系統(tǒng)上系統(tǒng)上運(yùn)行,而不同的計(jì)算機(jī)又有一定的差異。例如,我們有一個(gè)數(shù)據(jù)類(lèi)型,在Windows平臺(tái)中,應(yīng)該使用long類(lèi)型表示,而在其他平臺(tái)應(yīng)該使用float表示,這樣往往需要對(duì)源程序作必要的修改,這就降低了程序的通用性。可以用以下的條件編譯: #ifdef WINDOWS #define MYTYPE long #else #define MYTYPE float #endif 如果在Windows上編譯程序,則可以在程序的開(kāi)始加上 #define WINDOWS 這樣則編譯下面的命令行: #define MYTYPE long 如果在這組條件編譯命令之前曾出現(xiàn)以下命令行: #define WINDOWS 0 則預(yù)編譯后程序中的MYTYPE都用float代替。這樣,源程序可以不必作任何修改就可以用于不同類(lèi)型的計(jì)算機(jī)系統(tǒng)。當(dāng)然以上介紹的只是一種簡(jiǎn)單的情況,可以根據(jù)此思路設(shè)計(jì)出其它的條件編譯。 例如,在調(diào)試程序時(shí),常常希望輸出一些所需的信息,而在調(diào)試完成后不再輸出這些信息??梢栽谠闯绦蛑胁迦胍韵碌臈l件編譯段: #ifdef DEBUG print ("device_open(%p)\n", file); #endif 如果在它的前面有以下命令行: #define DEBUG 則在程序運(yùn)行時(shí)輸出file指針的值,以便調(diào)試分析。調(diào)試完成后只需將這個(gè)define命令行刪除即可。有人可能覺(jué)得不用條件編譯也可達(dá)此目的,即在調(diào)試時(shí)加一批printf語(yǔ)句,調(diào)試后一一將printf語(yǔ)句刪除去。的確,這是可以的。但是,當(dāng)調(diào)試時(shí)加的printf語(yǔ)句比較多時(shí),修改的工作量是很大的。用條件編譯,則不必一一刪改printf語(yǔ)句,只需刪除前面的一條“#define DEBUG”命令即可,這時(shí)所有的用DEBUG作標(biāo)識(shí)符的條件編譯段都使其中的printf語(yǔ)句不起作用,即起統(tǒng)一控制的作用,如同一個(gè)“開(kāi)關(guān)”一樣。 有時(shí)也采用下面的形式: #ifndef 標(biāo)識(shí)符 程序段1 #else 程序段2 #endif 只是第一行與第一種形式不同:將“ifdef”改為“ifndef”。它的作用是:若標(biāo)識(shí)符未被定義則編譯程序段1,否則編譯程序段2。這種形式與第一種形式的作用相反。 以上兩種形式用法差不多,根據(jù)需要任選一種,視方便而定。 還有一種形式,就是#if后面的是一個(gè)表達(dá)式,而不是一個(gè)簡(jiǎn)單的標(biāo)識(shí)符: #if 表達(dá)式 程序段1 #else 程序段2 #endif 它的作用是:當(dāng)指定的表達(dá)式值為真(非零)時(shí)就編譯程序段1,否則編譯程序段2??梢允孪冉o定一定條件,使程序在不同的條件下執(zhí)行不同的功能。 例如:輸入一行字母字符,根據(jù)需要設(shè)置條件編譯,使之能將字母全改為大寫(xiě)輸出,或全改為小寫(xiě)字母輸出。 #define LETTER 1 main() { char str[20]="C Language",c; int i=0; while((c=str[i])!='\0'){ i++; #if LETTER if(c>='a'&&c<='z') c=c-32; #else if(c>='A'&&c<='Z') c=c+32; #endif printf("%c",c); } } 運(yùn)行結(jié)果為:C LANGUAGE 現(xiàn)在先定義LETTER為1,這樣在預(yù)處理?xiàng)l件編譯命令時(shí),由于LETTER為真(非零),則對(duì)第一個(gè)if語(yǔ)句進(jìn)行編譯,運(yùn)行時(shí)使小寫(xiě)字母變大寫(xiě)。如果將程序第一行改為: #define LETTER 0 則在預(yù)處理時(shí),對(duì)第二個(gè)if語(yǔ)句進(jìn)行編譯處理,使大寫(xiě)字母變成小寫(xiě)字母(大寫(xiě)字母與相應(yīng)的小寫(xiě)字母的ASCII代碼差32)。此時(shí)運(yùn)行情況為: c language 有人會(huì)問(wèn):不用條件編譯命令而直接用if語(yǔ)句也能達(dá)到要求,用條件編譯命令有什么好處呢?的確,此問(wèn)題完全可以不用條件編譯處理,但那樣做目標(biāo)程序長(zhǎng)(因?yàn)樗姓Z(yǔ)句都編譯),而采用條件編譯,可以減少被編譯的語(yǔ)句,從而減少目標(biāo)的長(zhǎng)度。當(dāng)條件編譯段比較多時(shí),目標(biāo)程序長(zhǎng)度可以大大減少。
|