所謂預處理,就是指由預處理器處理的東東。編譯處理是由編譯器處理的東西,現(xiàn)代編譯器通常和預處理器集成在一起,所以大部分時間我們根本感覺不到預處理器的存在.由#define 所定義的都是由預處理器處理的,例如:
創(chuàng)新互聯(lián)公司服務(wù)項目包括舟曲網(wǎng)站建設(shè)、舟曲網(wǎng)站制作、舟曲網(wǎng)頁制作以及舟曲網(wǎng)絡(luò)營銷策劃等。多年來,我們專注于互聯(lián)網(wǎng)行業(yè),利用自身積累的技術(shù)優(yōu)勢、行業(yè)經(jīng)驗、深度合作伙伴關(guān)系等,向廣大中小型企業(yè)、政府機構(gòu)等提供互聯(lián)網(wǎng)行業(yè)的解決方案,舟曲網(wǎng)站推廣取得了明顯的社會效益與經(jīng)濟效益。目前,我們服務(wù)的客戶以成都為中心已經(jīng)輻射到舟曲省份的部分城市,未來相信會繼續(xù)擴大服務(wù)區(qū)域并繼續(xù)獲得客戶的支持與信任!
#define u16 unsigned short
typedef unsigned short u16;這兩個語句同樣是定義u16為無符號短整型,但是第一條語句是由預處理器來執(zhí)行替換操作的, 凡是遇到u16的地方都用unsigned short來替換(字符串里的u16不會被替換)。第二條語句就是由編譯器來完成的。通常預處理器不會檢查錯誤
第十一章 預處理概述
在前面各章中,已多次使用過以“#”號開頭的預處理命令。如包含命令# include,宏定義命令# define等。在源程序中這些命令都放在函數(shù)之外, 而且一般都放在源文件的前面,它們稱為預處理部分。所謂預處理是指在進行編譯的第一遍掃描(詞法掃描和語法分析)之前所作的工作。預處理是C語言的一個重要功能, 它由預處理程序負責完成。當對一個源文件進行編譯時, 系統(tǒng)將自動引用預處理程序?qū)υ闯绦蛑械念A處理部分作處理, 處理完畢自動進入對源程序的編譯。C語言提供了多種預處理功能,如宏定義、文件包含、 條件編譯等。合理地使用預處理功能編寫的程序便于閱讀、修改、 移植和調(diào)試,也有利于模塊化程序設(shè)計。本章介紹常用的幾種預處理功能。宏定義
在C語言源程序中允許用一個標識符來表示一個字符串, 稱為“宏”。被定義為“宏”的標識符稱為“宏名”。在編譯預處理時,對程序中所有出現(xiàn)的“宏名”,都用宏定義中的字符串去代換, 這稱為“宏代換”或“宏展開”。宏定義是由源程序中的宏定義命令完成的。 宏代換是由預處理程序自動完成的。在C語言中,“宏”分為有參數(shù)和無參數(shù)兩種。 下面分別討論這兩種“宏”的定義和調(diào)用。 無參宏定義
無參宏的宏名后不帶參數(shù)。其定義的一般形式為: #define 標識符 字符串 其中的“#”表示這是一條預處理命令。凡是以“#”開頭的均為預處理命令?!癲efine”為宏定義命令。 “標識符”為所定義的宏名。“字符串”可以是常數(shù)、表達式、格式串等。在前面介紹過的符號常量的定義就是一種無參宏定義。 此外,常對程序中反復使用的表達式進行宏定義。例如: # define M (y*y+3*y) 定義M表達式(y*y+3*y)。在編寫源程序時,所有的(y*y+3*y)都可由M代替,而對源程序作編譯時,將先由預處理程序進行宏代換,即用(y*y+3*y)表達式去置換所有的宏名M,然后再進行編譯。
#define M (y*y+3*y)
main(){
int s,y;
printf("input a number: ");
scanf("%d",y);
s=3*M+4*M+5*M;
printf("s=%d\n",s);
}
上例程序中首先進行宏定義,定義M表達式(y*y+3*y),在s= 3*M+4*M+5* M中作了宏調(diào)用。在預處理時經(jīng)宏展開后該語句變?yōu)椋簊=3*(y*y+3*y)+4(y*y+3*y)+5(y*y+3*y);但要注意的是,在宏定義中表達式(y*y+3*y)兩邊的括號不能少。否則會發(fā)生錯誤。
當作以下定義后: #difine M y*y+3*y在宏展開時將得到下述語句: s=3*y*y+3*y+4*y*y+3*y+5*y*y+3*y;這相當于; 3y?2+3y+4y?2+3y+5y?2+3y;顯然與原題意要求不符。計算結(jié)果當然是錯誤的。 因此在作宏定義時必須十分注意。應(yīng)保證在宏代換之后不發(fā)生錯誤。對于宏定義還要說明以下幾點:1. 宏定義是用宏名來表示一個字符串,在宏展開時又以該字符串取代宏名,這只是一種簡單的代換,字符串中可以含任何字符,可以是常數(shù),也可以是表達式,預處理程序?qū)λ蛔魅魏螜z查。如有錯誤,只能在編譯已被宏展開后的源程序時發(fā)現(xiàn)。2. 宏定義不是說明或語句,在行末不必加分號,如加上分號則連分號也一起置換。3. 宏定義必須寫在函數(shù)之外,其作用域為宏定義命令起到源程序結(jié) 束。如要終止其作用域可使用# undef命令,例如: # define PI 3.14159
main()
{
……
}
# undef PIPI的作用域
f1()
....表示PI只在main函數(shù)中有效,在f1中無效。
4. 宏名在源程序中若用引號括起來,則預處理程序不對其作宏代換。
#define OK 100
main()
{
printf("OK");
printf("\n");
}
上例中定義宏名OK表示100,但在printf語句中OK被引號括起來,因此不作宏代換。程序的運行結(jié)果為:OK這表示把“OK”當字符串處理。5. 宏定義允許嵌套,在宏定義的字符串中可以使用已經(jīng)定義的宏名。在宏展開時由預處理程序?qū)訉哟鷵Q。例如: #define PI 3.1415926
#define S PI*y*y /* PI是已定義的宏名*/對語句: printf("%f",s);在宏代換后變?yōu)椋?printf("%f",3.1415926*y*y);6. 習慣上宏名用大寫字母表示,以便于與變量區(qū)別。但也允許用小寫字母。7. 可用宏定義表示數(shù)據(jù)類型,使書寫方便。例如: #define STU struct stu在程序中可用STU作變量說明: STU body[5],*p;#define INTEGER int 在程序中即可用INTEGER作整型變量說明: INTEGER a,b; 應(yīng)注意用宏定義表示數(shù)據(jù)類型和用typedef定義數(shù)據(jù)說明符的區(qū)別。宏定義只是簡單的字符串代換,是在預處理完成的,而typedef是在編譯時處理的,它不是作簡單的代換, 而是對類型說明符重新命名。被命名的標識符具有類型定義說明的功能。請看下面的例子: #define PIN1 int* typedef (int*) PIN2;從形式上看這兩者相似, 但在實際使用中卻不相同。下面用PIN1,PIN2說明變量時就可以看出它們的區(qū)別: PIN1 a,b;在宏代換后變成 int *a,b;表示a是指向整型的指針變量,而b是整型變量。然而:PIN2 a,b;表示a,b都是指向整型的指針變量。因為PIN2是一個類型說明符。由這個例子可見,宏定義雖然也可表示數(shù)據(jù)類型, 但畢竟是作字符
代換。在使用時要分外小心,以避出錯。8. 對“輸出格式”作宏定義,可以減少書寫麻煩。例9.3 中就采用了這種方法。
#define P printf
#define D "%d\n"
#define F "%f\n"
main(){
int a=5, c=8, e=11;
float b=3.8, d=9.7, f=21.08;
P(D F,a,b);
P(D F,c,d);
P(D F,e,f);
}帶參宏定義C語言允許宏帶有參數(shù)。在宏定義中的參數(shù)稱為形式參數(shù), 在宏調(diào)用中的參數(shù)稱為實際參數(shù)。對帶參數(shù)的宏,在調(diào)用中,不僅要宏展開, 而且要用實參去代換形參。帶參宏定義的一般形式為: #define 宏名(形參表) 字符串 在字符串中含有各個形參。帶參宏調(diào)用的一般形式為: 宏名(實參表);
例如:
#define M(y) y*y+3*y /*宏定義*/
:
k=M(5); /*宏調(diào)用*/
: 在宏調(diào)用時,用實參5去代替形參y, 經(jīng)預處理宏展開后的語句
為: k=5*5+3*5
#define MAX(a,b) (ab)?a:b
main(){
int x,y,max;
printf("input two numbers: ");
scanf("%d%d",x,y);
max=MAX(x,y);
printf("max=%d\n",max);
}
上例程序的第一行進行帶參宏定義,用宏名MAX表示條件表達式(ab)?a:b,形參a,b均出現(xiàn)在條件表達式中。程序第七行max=MAX(x,
y)為宏調(diào)用,實參x,y,將代換形參a,b。宏展開后該語句為: max=(xy)?x:y;用于計算x,y中的大數(shù)。對于帶參的宏定義有以下問題需要說明:1. 帶參宏定義中,宏名和形參表之間不能有空格出現(xiàn)。
例如把: #define MAX(a,b) (ab)?a:b寫為: #define MAX (a,b) (ab)?a:b 將被認為是無參宏定義,宏名MAX代表字符串 (a,b)(ab)?a:b。
宏展開時,宏調(diào)用語句: max=MAX(x,y);將變?yōu)椋?max=(a,b)(ab)?a:b(x,y);這顯然是錯誤的。2. 在帶參宏定義中,形式參數(shù)不分配內(nèi)存單元,因此不必作類型定義。而宏調(diào)用中的實參有具體的值。要用它們?nèi)ゴ鷵Q形參,因此必須作類型說明。這是與函數(shù)中的情況不同的。在函數(shù)中,形參和實參是兩個不同的量,各有自己的作用域,調(diào)用時要把實參值賦予形參,進行“值傳遞”。而在帶參宏中,只是符號代換,不存在值傳遞的問題。3. 在宏定義中的形參是標識符,而宏調(diào)用中的實參可以是表達式。
#define SQ(y) (y)*(y)
main(){
int a,sq;
printf("input a number: ");
scanf("%d",a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
上例中第一行為宏定義,形參為y。程序第七行宏調(diào)用中實參為a+1,是一個表達式,在宏展開時,用a+1代換y,再用(y)*(y) 代換SQ,得到如下語句: sq=(a+1)*(a+1); 這與函數(shù)的調(diào)用是不同的, 函數(shù)調(diào)用時要把實參表達式的值求出來再賦予形參。 而宏代換中對實參表達式不作計算直接地照原樣代換。4. 在宏定義中,字符串內(nèi)的形參通常要用括號括起來以避免出錯。 在上例中的宏定義中(y)*(y)表達式的y都用括號括起來,因此結(jié)果是正確的。如果去掉括號,把程序改為以下形式:
#define SQ(y) y*y
main(){
int a,sq;
printf("input a number: ");
scanf("%d",a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
運行結(jié)果為:input a number:3
sq=7 同樣輸入3,但結(jié)果卻是不一樣的。問題在哪里呢? 這是由于代換只作符號代換而不作其它處理而造成的。 宏代換后將得到以下語句: sq=a+1*a+1; 由于a為3故sq的值為7。這顯然與題意相違,因此參數(shù)兩邊的括號是不能少的。即使在參數(shù)兩邊加括號還是不夠的,請看下面程序:
#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); 運行本程序如輸入值仍為3時,希望結(jié)果為10。但實際運行的結(jié)果如下:input a number:3 sq=160為什么會得這樣的結(jié)果呢?分析宏調(diào)用語句,在宏代換之后變?yōu)椋?sq=160/(a+1)*(a+1);a為3時,由于“/”和“*”運算符優(yōu)先級和結(jié)合性相同, 則先作160/(3+1)得40,再作40*(3+1)最后得160。為了得到正確答案應(yī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);
}
以上討論說明,對于宏定義不僅應(yīng)在參數(shù)兩側(cè)加括號, 也應(yīng)在整個字符串外加括號。5. 帶參的宏和帶參函數(shù)很相似,但有本質(zhì)上的不同,除上面已談到的各點外,把同一表達式用函數(shù)處理與用宏處理兩者的結(jié)果有可能是不同的。main(){
int i=1;
while(i=5)
printf("%d\n",SQ(i++));
}
SQ(int y)
{
return((y)*(y));
}#define SQ(y) ((y)*(y))
main(){
int i=1;
while(i=5)
printf("%d\n",SQ(i++));
}
在上例中函數(shù)名為SQ,形參為Y,函數(shù)體表達式為((y)*(y))。在例9.6中宏名為SQ,形參也為y,字符串表達式為(y)*(y))。 兩例是相同的。例9.6的函數(shù)調(diào)用為SQ(i++),例9.7的宏調(diào)用為SQ(i++),實參也是相同的。從輸出結(jié)果來看,卻大不相同。分析如下:在例9.6中,函數(shù)調(diào)用是把實參i值傳給形參y后自增1。 然后輸出函數(shù)值。因而要循環(huán)5次。輸出1~5的平方值。而在例9.7中宏調(diào)用時,只作代換。SQ(i++)被代換為((i++)*(i++))。在第一次循環(huán)時,由于i等于1,其計算過程為:表達式中前一個i初值為1,然后i自增1變?yōu)?,因此表達式中第2個i初值為2,兩相乘的結(jié)果也為2,然后i值再自增1,得3。在第二次循環(huán)時,i值已有初值為3,因此表達式中前一個i為3,后一個i為4, 乘積為12,然后i再自增1變?yōu)?。進入第三次循環(huán),由于i 值已為5,所以這將是最后一次循環(huán)。計算表達式的值為5*6等于30。i值再自增1變?yōu)?,不再滿足循環(huán)條件,停止循環(huán)。從以上分析可以看出函數(shù)調(diào)用和宏調(diào)用二者在形式上相似, 在本質(zhì)上是完全不同的。6. 宏定義也可用來定義多個語句,在宏調(diào)用時,把這些語句又代換到源程序內(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個賦值語句,4 個形參分別為4個賦值符左部的變量。在宏調(diào)用時,把4 個語句展開并用實參代替形參。使計算結(jié)果送入實參之中。文件包含文件包含是C預處理程序的另一個重要功能。文件包含命令行的一般形式為: #include"文件名" 在前面我們已多次用此命令包含過庫函數(shù)的頭文件。例如:
#include"stdio.h"
#include"math.h"
文件包含命令的功能是把指定的文件插入該命令行位置取代該命令行, 從而把指定的文件和當前的源程序文件連成一個源文件。在程序設(shè)計中,文件包含是很有用的。 一個大的程序可以分為多個模塊,由多個程序員分別編程。 有些公用的符號常量或宏定義等可單獨組成一個文件, 在其它文件的開頭用包含命令包含該文件即可使用。這樣,可避免在每個文件開頭都去書寫那些公用量, 從而節(jié)省時間,并減少出錯。對文件包含命令還要說明以下幾點:
1. 包含命令中的文件名可以用雙引號括起來,也可以用尖括號括起來。例如以下寫法都是允許的: #include"stdio.h" #includemath.h 但是這兩種形式是有區(qū)別的:使用尖括號表示在包含文件目錄中去查找(包含目錄是由用戶在設(shè)置環(huán)境時設(shè)置的), 而不在源文件目錄去查找; 使用雙引號則表示首先在當前的源文件目錄中查找,若未找到才到包含目錄中去查找。 用戶編程時可根據(jù)自己文件所在的目錄來選擇某一種命令形式。2. 一個include命令只能指定一個被包含文件, 若有多個文件要包含,則需用多個include命令。3. 文件包含允許嵌套,即在一個被包含的文件中又可以包含另一個文件。條件編譯預處理程序提供了條件編譯的功能。 可以按不同的條件去編譯不同的程序部分,因而產(chǎn)生不同的目標代碼文件。 這對于程序的移植和調(diào)試是很有用的。 條件編譯有三種形式,下面分別介紹:
1. 第一種形式:
#ifdef 標識符
程序段1
#else
程序段2
#endif
它的功能是,如果標識符已被 #define命令定義過則對程序段1進行編譯;否則對程序段2進行編譯。如果沒有程序段2(它為空),本格式中的#else可以沒有, 即可以寫為:
#ifdef 標識符
程序段 #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行插入了條件編譯預處理命令, 因此要根據(jù)NUM是否被定義過來決定編譯那一個printf語句。而在程序的第一行已對NUM作過宏定義,因此應(yīng)對第一個printf語句作編譯故運行結(jié)果是輸出了學號和成績。在程序的第一行宏定義中,定義NUM表示字符串OK,其實也可以為任何字符串,甚至不給出任何字符串,寫為: #define NUM 也具有同樣的意義。 只有取消程序的第一行才會去編譯第二個printf語句。讀者可上機試作。2. 第二種形式:
#ifndef 標識符
程序段1
#else
程序段2
#endif
與第一種形式的區(qū)別是將“ifdef”改為“ifndef”。它的功能是,如果標識符未被#define命令定義過則對程序段1進行編譯, 否則對程序段2進行編譯。這與第一種形式的功能正相反。3. 第三種形式:
#if 常量表達式
程序段1
#else
程序段2
#endif
它的功能是,如常量表達式的值為真(非0),則對程序段1 進行編譯,否則對程序段2進行編譯。因此可以使程序在不同條件下,完成不同的功能
#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,因此在條件編譯時,常量表達式的值為真, 故計算并輸出圓面積。上面介紹的條件編譯當然也可以用條件語句來實現(xiàn)。 但是用條件語句將會對整個源程序進行編譯,生成的目標代碼程序很長,而采用條件編譯,則根據(jù)條件只編譯其中的程序段1或程序段2, 生成的目標程序較短。如果條件選擇的程序段很長, 采用條件編譯的方法是十分必要的。
C語言的編譯過程分為:預處理-匯編-編譯-鏈接 的過程。
預處理的意思是,為下一步匯編(生成匯編)做準備。包括
頭文件包含,即#include包含的文件添加進來
宏展開,即#define 定義的宏在程序中代入進去
條件編譯,選擇具體用來編譯的語句
預處理器的主要作用就是把通過預處理的內(nèi)建功能對一個資源進行等價替
換,最常見的預處理有:文件包含,條件編譯、布局控制和宏替換4種
文件包含:#include
是一種最為常見的預處理,主要是做為文件的引用組合
源程序正文。
條件編譯:#if,#ifndef,#ifdef,#endif,#undef等也是比較常見的預處理,
主要是進行編譯時
進行有選擇的挑選,
注釋掉一些指定的代碼,
以達到版本控制、
防止對文件重復包含的功能。
布局控制:
#progma,這也是我們應(yīng)用預處理的一個重要方面,主要功能是
為編譯程序提供非常規(guī)的控制流信息。
宏替換:
#define,這是最常見的用法,它可以定義符號常量、函數(shù)功能、
重新命名、字符串的拼接等各種功能
第十一章 預處理概述
在前面各章中,已多次使用過以“#”號開頭的預處理命令。如包含命令# include,宏定義命令# define等。在源程序中這些命令都放在函數(shù)之外, 而且一般都放在源文件的前面,它們稱為預處理部分。所謂預處理是指在進行編譯的第一遍掃描(詞法掃描和語法分析)之前所作的工作。預處理是C語言的一個重要功能, 它由預處理程序負責完成。當對一個源文件進行編譯時, 系統(tǒng)將自動引用預處理程序?qū)υ闯绦蛑械念A處理部分作處理, 處理完畢自動進入對源程序的編譯。C語言提供了多種預處理功能,如宏定義、文件包含、 條件編譯等。合理地使用預處理功能編寫的程序便于閱讀、修改、 移植和調(diào)試,也有利于模塊化程序設(shè)計。本章介紹常用的幾種預處理功能。宏定義
在C語言源程序中允許用一個標識符來表示一個字符串, 稱為“宏”。被定義為“宏”的標識符稱為“宏名”。在編譯預處理時,對程序中所有出現(xiàn)的“宏名”,都用宏定義中的字符串去代換, 這稱為“宏代換”或“宏展開”。宏定義是由源程序中的宏定義命令完成的。 宏代換是由預處理程序自動完成的。在C語言中,“宏”分為有參數(shù)和無參數(shù)兩種。 下面分別討論這兩種“宏”的定義和調(diào)用。 無參宏定義
無參宏的宏名后不帶參數(shù)。其定義的一般形式為: #define 標識符 字符串 其中的“#”表示這是一條預處理命令。凡是以“#”開頭的均為預處理命令。“define”為宏定義命令。 “標識符”為所定義的宏名?!白址笨梢允浅?shù)、表達式、格式串等。在前面介紹過的符號常量的定義就是一種無參宏定義。 此外,常對程序中反復使用的表達式進行宏定義。例如: # define M (y*y+3*y) 定義M表達式(y*y+3*y)。在編寫源程序時,所有的(y*y+3*y)都可由M代替,而對源程序作編譯時,將先由預處理程序進行宏代換,即用(y*y+3*y)表達式去置換所有的宏名M,然后再進行編譯。
#define M (y*y+3*y)
main(){
int s,y;
printf("input a number: ");
scanf("%d",y);
s=3*M+4*M+5*M;
printf("s=%d\n",s);
}
上例程序中首先進行宏定義,定義M表達式(y*y+3*y),在s= 3*M+4*M+5* M中作了宏調(diào)用。在預處理時經(jīng)宏展開后該語句變?yōu)椋簊=3*(y*y+3*y)+4(y*y+3*y)+5(y*y+3*y);但要注意的是,在宏定義中表達式(y*y+3*y)兩邊的括號不能少。否則會發(fā)生錯誤。
當作以下定義后: #difine M y*y+3*y在宏展開時將得到下述語句: s=3*y*y+3*y+4*y*y+3*y+5*y*y+3*y;這相當于; 3y?2+3y+4y?2+3y+5y?2+3y;顯然與原題意要求不符。計算結(jié)果當然是錯誤的。 因此在作宏定義時必須十分注意。應(yīng)保證在宏代換之后不發(fā)生錯誤。對于宏定義還要說明以下幾點:1. 宏定義是用宏名來表示一個字符串,在宏展開時又以該字符串取代宏名,這只是一種簡單的代換,字符串中可以含任何字符,可以是常數(shù),也可以是表達式,預處理程序?qū)λ蛔魅魏螜z查。如有錯誤,只能在編譯已被宏展開后的源程序時發(fā)現(xiàn)。2. 宏定義不是說明或語句,在行末不必加分號,如加上分號則連分號也一起置換。3. 宏定義必須寫在函數(shù)之外,其作用域為宏定義命令起到源程序結(jié) 束。如要終止其作用域可使用# undef命令,例如: # define PI 3.14159
main()
{
……
}
# undef PIPI的作用域
f1()
....表示PI只在main函數(shù)中有效,在f1中無效。
4. 宏名在源程序中若用引號括起來,則預處理程序不對其作宏代換。
#define OK 100
main()
{
printf("OK");
printf("\n");
}
上例中定義宏名OK表示100,但在printf語句中OK被引號括起來,因此不作宏代換。程序的運行結(jié)果為:OK這表示把“OK”當字符串處理。5. 宏定義允許嵌套,在宏定義的字符串中可以使用已經(jīng)定義的宏名。在宏展開時由預處理程序?qū)訉哟鷵Q。例如: #define PI 3.1415926
#define S PI*y*y /* PI是已定義的宏名*/對語句: printf("%f",s);在宏代換后變?yōu)椋?printf("%f",3.1415926*y*y);6. 習慣上宏名用大寫字母表示,以便于與變量區(qū)別。但也允許用小寫字母。7. 可用宏定義表示數(shù)據(jù)類型,使書寫方便。例如: #define STU struct stu在程序中可用STU作變量說明: STU body[5],*p;#define INTEGER int在程序中即可用INTEGER作整型變量說明: INTEGER a,b; 應(yīng)注意用宏定義表示數(shù)據(jù)類型和用typedef定義數(shù)據(jù)說明符的區(qū)別。宏定義只是簡單的字符串代換,是在預處理完成的,而typedef是在編譯時處理的,它不是作簡單的代換, 而是對類型說明符重新命名。被命名的標識符具有類型定義說明的功能。請看下面的例子: #define PIN1 int*typedef (int*) PIN2;從形式上看這兩者相似, 但在實際使用中卻不相同。下面用PIN1,PIN2說明變量時就可以看出它們的區(qū)別: PIN1 a,b;在宏代換后變成 int *a,b;表示a是指向整型的指針變量,而b是整型變量。然而:PIN2 a,b;表示a,b都是指向整型的指針變量。因為PIN2是一個類型說明符。由這個例子可見,宏定義雖然也可表示數(shù)據(jù)類型, 但畢竟是作字符
代換。在使用時要分外小心,以避出錯。8. 對“輸出格式”作宏定義,可以減少書寫麻煩。例9.3 中就采用了這種方法。
#define P printf
#define D "%d\n"
#define F "%f\n"
main(){
int a=5, c=8, e=11;
float b=3.8, d=9.7, f=21.08;
P(D F,a,b);
P(D F,c,d);
P(D F,e,f);
}帶參宏定義C語言允許宏帶有參數(shù)。在宏定義中的參數(shù)稱為形式參數(shù), 在宏調(diào)用中的參數(shù)稱為實際參數(shù)。對帶參數(shù)的宏,在調(diào)用中,不僅要宏展開, 而且要用實參去代換形參。帶參宏定義的一般形式為: #define 宏名(形參表) 字符串 在字符串中含有各個形參。帶參宏調(diào)用的一般形式為: 宏名(實參表);
例如:
#define M(y) y*y+3*y /*宏定義*/
:
k=M(5); /*宏調(diào)用*/
: 在宏調(diào)用時,用實參5去代替形參y, 經(jīng)預處理宏展開后的語句
為: k=5*5+3*5
#define MAX(a,b) (ab)?a:b
main(){
int x,y,max;
printf("input two numbers: ");
scanf("%d%d",x,y);
max=MAX(x,y);
printf("max=%d\n",max);
}
上例程序的第一行進行帶參宏定義,用宏名MAX表示條件表達式(ab)?a:b,形參a,b均出現(xiàn)在條件表達式中。程序第七行max=MAX(x,
y)為宏調(diào)用,實參x,y,將代換形參a,b。宏展開后該語句為: max=(xy)?x:y;用于計算x,y中的大數(shù)。對于帶參的宏定義有以下問題需要說明:1. 帶參宏定義中,宏名和形參表之間不能有空格出現(xiàn)。
例如把: #define MAX(a,b) (ab)?a:b寫為: #define MAX (a,b) (ab)?a:b 將被認為是無參宏定義,宏名MAX代表字符串 (a,b)(ab)?a:b。
宏展開時,宏調(diào)用語句: max=MAX(x,y);將變?yōu)椋?max=(a,b)(ab)?a:b(x,y);這顯然是錯誤的。2. 在帶參宏定義中,形式參數(shù)不分配內(nèi)存單元,因此不必作類型定義。而宏調(diào)用中的實參有具體的值。要用它們?nèi)ゴ鷵Q形參,因此必須作類型說明。這是與函數(shù)中的情況不同的。在函數(shù)中,形參和實參是兩個不同的量,各有自己的作用域,調(diào)用時要把實參值賦予形參,進行“值傳遞”。而在帶參宏中,只是符號代換,不存在值傳遞的問題。3. 在宏定義中的形參是標識符,而宏調(diào)用中的實參可以是表達式。
#define SQ(y) (y)*(y)
main(){
int a,sq;
printf("input a number: ");
scanf("%d",a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
上例中第一行為宏定義,形參為y。程序第七行宏調(diào)用中實參為a+1,是一個表達式,在宏展開時,用a+1代換y,再用(y)*(y) 代換SQ,得到如下語句: sq=(a+1)*(a+1); 這與函數(shù)的調(diào)用是不同的, 函數(shù)調(diào)用時要把實參表達式的值求出來再賦予形參。 而宏代換中對實參表達式不作計算直接地照原樣代換。4. 在宏定義中,字符串內(nèi)的形參通常要用括號括起來以避免出錯。 在上例中的宏定義中(y)*(y)表達式的y都用括號括起來,因此結(jié)果是正確的。如果去掉括號,把程序改為以下形式:
#define SQ(y) y*y
main(){
int a,sq;
printf("input a number: ");
scanf("%d",a);
sq=SQ(a+1);
printf("sq=%d\n",sq);
}
運行結(jié)果為:input a number:3
sq=7 同樣輸入3,但結(jié)果卻是不一樣的。問題在哪里呢? 這是由于代換只作符號代換而不作其它處理而造成的。 宏代換后將得到以下語句: sq=a+1*a+1; 由于a為3故sq的值為7。這顯然與題意相違,因此參數(shù)兩邊的括號是不能少的。即使在參數(shù)兩邊加括號還是不夠的,請看下面程序:
#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); 運行本程序如輸入值仍為3時,希望結(jié)果為10。但實際運行的結(jié)果如下:input a number:3sq=160為什么會得這樣的結(jié)果呢?分析宏調(diào)用語句,在宏代換之后變?yōu)椋?sq=160/(a+1)*(a+1);a為3時,由于“/”和“*”運算符優(yōu)先級和結(jié)合性相同, 則先作160/(3+1)得40,再作40*(3+1)最后得160。為了得到正確答案應(yī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);
}
以上討論說明,對于宏定義不僅應(yīng)在參數(shù)兩側(cè)加括號, 也應(yīng)在整個字符串外加括號。5. 帶參的宏和帶參函數(shù)很相似,但有本質(zhì)上的不同,除上面已談到的各點外,把同一表達式用函數(shù)處理與用宏處理兩者的結(jié)果有可能是不同的。main(){
int i=1;
while(i=5)
printf("%d\n",SQ(i++));
}
SQ(int y)
{
return((y)*(y));
}#define SQ(y) ((y)*(y))
main(){
int i=1;
while(i=5)
printf("%d\n",SQ(i++));
}
在上例中函數(shù)名為SQ,形參為Y,函數(shù)體表達式為((y)*(y))。在例9.6中宏名為SQ,形參也為y,字符串表達式為(y)*(y))。 兩例是相同的。例9.6的函數(shù)調(diào)用為SQ(i++),例9.7的宏調(diào)用為SQ(i++),實參也是相同的。從輸出結(jié)果來看,卻大不相同。分析如下:在例9.6中,函數(shù)調(diào)用是把實參i值傳給形參y后自增1。 然后輸出函數(shù)值。因而要循環(huán)5次。輸出1~5的平方值。而在例9.7中宏調(diào)用時,只作代換。SQ(i++)被代換為((i++)*(i++))。在第一次循環(huán)時,由于i等于1,其計算過程為:表達式中前一個i初值為1,然后i自增1變?yōu)?,因此表達式中第2個i初值為2,兩相乘的結(jié)果也為2,然后i值再自增1,得3。在第二次循環(huán)時,i值已有初值為3,因此表達式中前一個i為3,后一個i為4, 乘積為12,然后i再自增1變?yōu)?。進入第三次循環(huán),由于i 值已為5,所以這將是最后一次循環(huán)。計算表達式的值為5*6等于30。i值再自增1變?yōu)?,不再滿足循環(huán)條件,停止循環(huán)。從以上分析可以看出函數(shù)調(diào)用和宏調(diào)用二者在形式上相似, 在本質(zhì)上是完全不同的。6. 宏定義也可用來定義多個語句,在宏調(diào)用時,把這些語句又代換到源程序內(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個賦值語句,4 個形參分別為4個賦值符左部的變量。在宏調(diào)用時,把4 個語句展開并用實參代替形參。使計算結(jié)果送入實參之中。文件包含文件包含是C預處理程序的另一個重要功能。文件包含命令行的一般形式為: #include"文件名" 在前面我們已多次用此命令包含過庫函數(shù)的頭文件。例如:
#include"stdio.h"
#include"math.h"
文件包含命令的功能是把指定的文件插入該命令行位置取代該命令行, 從而把指定的文件和當前的源程序文件連成一個源文件。在程序設(shè)計中,文件包含是很有用的。 一個大的程序可以分為多個模塊,由多個程序員分別編程。 有些公用的符號常量或宏定義等可單獨組成一個文件, 在其它文件的開頭用包含命令包含該文件即可使用。這樣,可避免在每個文件開頭都去書寫那些公用量, 從而節(jié)省時間,并減少出錯。對文件包含命令還要說明以下幾點:
1. 包含命令中的文件名可以用雙引號括起來,也可以用尖括號括起來。例如以下寫法都是允許的: #include"stdio.h"#includemath.h 但是這兩種形式是有區(qū)別的:使用尖括號表示在包含文件目錄中去查找(包含目錄是由用戶在設(shè)置環(huán)境時設(shè)置的), 而不在源文件目錄去查找; 使用雙引號則表示首先在當前的源文件目錄中查找,若未找到才到包含目錄中去查找。 用戶編程時可根據(jù)自己文件所在的目錄來選擇某一種命令形式。2. 一個include命令只能指定一個被包含文件, 若有多個文件要包含,則需用多個include命令。3. 文件包含允許嵌套,即在一個被包含的文件中又可以包含另一個文件。條件編譯預處理程序提供了條件編譯的功能。 可以按不同的條件去編譯不同的程序部分,因而產(chǎn)生不同的目標代碼文件。 這對于程序的移植和調(diào)試是很有用的。 條件編譯有三種形式,下面分別介紹:
1. 第一種形式:
#ifdef 標識符
程序段1
#else
程序段2
#endif
它的功能是,如果標識符已被 #define命令定義過則對程序段1進行編譯;否則對程序段2進行編譯。如果沒有程序段2(它為空),本格式中的#else可以沒有, 即可以寫為:
#ifdef 標識符
程序段 #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行插入了條件編譯預處理命令, 因此要根據(jù)NUM是否被定義過來決定編譯那一個printf語句。而在程序的第一行已對NUM作過宏定義,因此應(yīng)對第一個printf語句作編譯故運行結(jié)果是輸出了學號和成績。在程序的第一行宏定義中,定義NUM表示字符串OK,其實也可以為任何字符串,甚至不給出任何字符串,寫為: #define NUM 也具有同樣的意義。 只有取消程序的第一行才會去編譯第二個printf語句。讀者可上機試作。2. 第二種形式:
#ifndef 標識符
程序段1
#else
程序段2
#endif
與第一種形式的區(qū)別是將“ifdef”改為“ifndef”。它的功能是,如果標識符未被#define命令定義過則對程序段1進行編譯, 否則對程序段2進行編譯。這與第一種形式的功能正相反。3. 第三種形式:
#if 常量表達式
程序段1
#else
程序段2
#endif
它的功能是,如常量表達式的值為真(非0),則對程序段1 進行編譯,否則對程序段2進行編譯。因此可以使程序在不同條件下,完成不同的功能
#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,因此在條件編譯時,常量表達式的值為真, 故計算并輸出圓面積。上面介紹的條件編譯當然也可以用條件語句來實現(xiàn)。 但是用條件語句將會對整個源程序進行編譯,生成的目標代碼程序很長,而采用條件編譯,則根據(jù)條件只編譯其中的程序段1或程序段2, 生成的目標程序較短。如果條件選擇的程序段很長, 采用條件編譯的方法是十分必要的。
include的作用:#include叫做“文件包含” /?include?語句包含并運行指定文件。功能:它的功能是用來指定文件的全部內(nèi)容替換程序中的命令行,從而使指定的文件與當前源文件連成一個源文件書寫格式:#include 文件名 //表示編譯系統(tǒng)根據(jù)系統(tǒng)頭文件存放的目錄路徑去搜索系統(tǒng)頭文件,而不是在源文件目錄去查找 #include "文件名" //表示編譯系統(tǒng)首先在當前的源文件目錄中查找,若未找到才根據(jù)系統(tǒng)的頭文件存放的目錄路徑去搜索系統(tǒng)頭文件//系統(tǒng)定義的頭文件通常使用尖括號;用戶自定義的頭文件通常使用雙引號。 //一般來說,如果為調(diào)用庫函數(shù)而用#include命令來包含相關(guān)的頭文件,則用尖括號,以節(jié)約查找時間。 //如果要包含的是用戶自己編寫的文件(這種文件一般都在用戶當前的目錄中),一般用雙撇號。 //若文件不在當前目錄中,在雙撇號內(nèi)應(yīng)該給出文件路徑(如#include"C:\wang\file2.h") ----------------- 1.#define 的作用 在C或C++語言源程序中允許用一個標識符來表示一個字符串,稱為“宏”。被定義為“宏”的標識符稱為“宏名”。在編譯預處理時,對程序中所有出現(xiàn)的“宏名”,都用宏定義中的字符串去代換,這稱為“宏代換”或“宏展開”。宏定義是由源程序中的宏定義命令完成的。宏代換是由預處理程序自動完成的。 在C或C++語言中,“宏”分為有參數(shù)和無參數(shù)兩種。 2. 無參宏定義 無參宏的宏名后不帶參數(shù)。 其定義的一般形式為: #define 標識符 字符串 其中的“#”表示這是一條預處理命令。凡是以“#”開頭的均為預處理命令?!癲efine”為宏定義命令?!皹俗R符”為所定義的宏名?!白址笨梢允浅?shù)、表達式、格式串等。 例如: #define M (a+b) 它的作用是指定標識符M來代替表達式(a+b)。在編寫源程序時,所有的(a+b)都可由M代替,而對源程序作編譯時,將先由預處理程序進行宏代換,即用(a+b)表達式去置換所有的宏名M,然后再進行編譯。 程序1: #define M (a+b) main(){ int s,y; printf("input a number: "); scanf("%d",y); s=M*M; printf("s=%d\n",s); } 上例程序中首先進行宏定義,定義M來替代表達式(a+b),在 s= M * M 中作了宏調(diào)用。在預處理時經(jīng)宏展開后該語句變?yōu)椋?S=(a+b)*(a+b) 但要注意的是,在宏定義中表達式(a+b)兩邊的括號不能少。否則會發(fā)生錯誤。 如當作以下定義后:#difine M (a)+(b) 在宏展開時將得到下述語句:S= (a)+(b)*(a)+(b) 對于宏定義還要說明以下幾點: 1. 宏定義是用宏名來表示一個字符串,在宏展開時又以該字符串取代宏名,這只是一種簡單的代換,字符串中可以含任何字符,可以是常數(shù),也可以是表達式,預處理程序?qū)λ蛔魅魏螜z查。如有錯誤,只能在編譯已被宏展開后的源程序時發(fā)現(xiàn)。 2. 宏定義不是說明或語句,在行末不必加分號,如加上分號則連分號也一起置換。 3. 宏定義必須寫在函數(shù)之外,其作用域為宏定義命令起到源程序結(jié)束。如要終止其作用域可使用#undef命令。 3. 帶參宏定義 c語言允許宏帶有參數(shù)。在宏定義中的參數(shù)稱為形式參數(shù),在宏調(diào)用中的參數(shù)稱為實際參數(shù)。對帶參數(shù)的宏,在調(diào)用中,不僅要宏展開,而且要用實參去代換形參。 帶參宏定義的一般形式為: #define 宏名(形參表) 字符串 在字符串中含有各個形參。 帶參宏調(diào)用的一般形式為: 宏名(形參表) 例如: #define M(y) ((y)*(y)+3*(y)) /*宏定義*/ ....k=M(5); /*宏調(diào)用*/ .... 在宏調(diào)用時,用實參5去代替形參y,經(jīng)預處理宏展開后的語句為: k=5*5+3*5 程序2: #define MAX(a,b) (ab)?a:b main(){ int x,y,max; printf("input two numbers: "); scanf("%d%d",x,y); max=MAX(x,y); printf("max=%d\n",max); } 上例程序的第一行進行帶參宏定義,用宏名MAX表示條件表達式(ab)?a:b,形參a,b均出現(xiàn)在條件表達式中。程序第七行max=MAX(x,y)為宏調(diào)用,實參x,y,將代換形參a,b。宏展開后該語句為: max=(xy)?x:y; 用于計算x,y中的大數(shù)。
網(wǎng)站欄目:C語言函數(shù)預處理意思 C語言什么是預處理
本文鏈接:http://jinyejixie.com/article46/dochohg.html
成都網(wǎng)站建設(shè)公司_創(chuàng)新互聯(lián),為您提供虛擬主機、ChatGPT、面包屑導航、App開發(fā)、搜索引擎優(yōu)化、網(wǎng)站收錄
聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權(quán)請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)