C語言宏定義的使用
時間:2021-09-03 10:08:23
手機看文章
掃描二維碼
隨時隨地手機看文章
[導讀]1概述在工程規(guī)模較小,不是很復雜,與硬件結合緊密,要求移植性的時候,可采用宏定義簡化編程,增強程序可讀性。當宏作為常量使用時,C程序員習慣在名字中只使用大寫字母。但是并沒有如何將用于其他目的的宏大寫的統(tǒng)一做法。由于宏(特別是帶參數(shù)的宏)可能是程序中錯誤的來源,所以一些程序員更喜歡...
1 概述
在工程規(guī)模較小,不是很復雜,與硬件結合緊密,要求移植性的時候,可采用宏定義簡化編程,增強程序可讀性。當宏作為常量使用時,C程序員習慣在名字中只使用大寫字母。但是并沒有如何將用于其他目的的宏大寫的統(tǒng)一做法。由于宏(特別是帶參數(shù)的宏)可能是程序中錯誤的來源,所以一些程序員更喜歡使用大寫字母來引起注意。- 簡單宏定義
#define 標識符 字符串
// 不帶參數(shù)的宏定義
#define MAX 10
注意:不要在宏定義中放置任何額外的符號,比如"="或者尾部加";"使用#define來為常量命名一些優(yōu)點:- 程序會更易讀。一個認真選擇的名字可以幫助讀者理解常量的意義;
- 程序會更易于修改。我們僅需要改變一個宏定義,就可以改變整個程序中出現(xiàn)的所有該常量的值;
- 可以幫助避免前后不一致或鍵盤輸入錯誤;
- 控制條件編譯;
- 可以對C語法做小的修改;
- 帶參數(shù)的宏
#define <宏名>(<參數(shù)列表>) <宏體>
注意參數(shù)列表中的參數(shù)必須是有效的c標識符,同時以,分隔算符優(yōu)先級問題:#define COUNT(M) M*M
int x=5;
print(COUNT(x 1));
print(COUNT( X));
//結果輸出:11 和42 而不是函數(shù)的輸出36
注意:- 預編譯器只是進行簡單的文本替換,COUNT(x 1)被替換成COUNT(x 1x 1),5 15 1=11,而不是36
- CUNT( x)被替換成 x* x即為6*7=42,而不是想要的6*6=36,連續(xù)前置自加加兩次
- 用括號將整個替換文本及每個參數(shù)用括號括起來
print(COUNT((x 1));
- 即便是加上括號也不能解決第二種情況,所以解決辦法是盡量不使用 ,-等符號;
#define foo(x) bar(x); baz(x)
假設這樣調用:if (!feral)
foo(wolf);
將被宏擴展為:if (!feral)
bar(wolf);
baz(wolf);
==baz(wolf);==,不在判斷條件中,顯而易見,這是錯誤。如果用大括號將其包起來依然會有問題,例如#define foo(x) { bar(x); baz(x); }
if (!feral)
foo(wolf);
else
bin(wolf);
判斷語言被擴展成:if (!feral) {
bar(wolf);
baz(wolf);
}>> ; <<
else
bin(wolf);
==else==將不會被執(zhí)行解決方法:通過==do{…}while(0)#define foo(x) do{ bar(x); baz(x); }while(0)
if (!feral)
foo(wolf);
else
bin(wolf);
被擴展成:#define foo(x) do{ bar(x); baz(x); }while(0)
if (!feral)
do{ bar(x); baz(x); }while(0);
else
bin(wolf);
注意:使用do{…}while(0)構造后的宏定義不會受到大括號、分號等的影響,總是會按你期望的方式調用運行。- #運算符
#define TEST(param) #param
char *pStr=TEST(123);
printf("pSrt=%s\n",pStr);
//輸出結果為字符 ”123“
- ##運算符
#define TEST(param1,param2) (param1##param2)
int num =TEST(13,59);
printf("num=%d\n",num);
//輸出結果為:num=1359
- VA_ARGS
# define PR(...) printf(_VA_ARGS_)
2 PR("hello world\n");
3
4 輸出結果:hello world
2 一些建議
- 雖然宏定義很靈活,并且通過彼此結合可以產(chǎn)生許多變形用法,但是C /C程序員不要定義很復雜的宏,宏定義應該簡單而清晰。
- 宏名采用大寫字符組成的單詞或其縮寫序列,并在各單詞之間使用“_”分隔。
- 如果需要公布某個宏,那么該宏定義應當放置在頭文件中,否則放置在實現(xiàn)文件(.cpp)的頂部。
- 不要使用宏來定義新類型名,應該使用typedef,否則容易造成錯誤。
- 給宏添加注釋時請使用塊注釋(/* */),而不要使用行注釋。因為有些編譯器可能會把宏后面的行注釋理解為宏體的一部分。
- 盡量使用const取代宏來定義符號常量。
- 對于較長的使用頻率較高的重復代碼片段,建議使用函數(shù)或模板而不要使用帶參數(shù)的宏定義;而對于較短的重復代碼片段,可以使用帶參數(shù)的宏定義,這不僅是出于類型安全的考慮,而且也是優(yōu)化與折衷的體現(xiàn)。
- 盡量避免在局部范圍內(如函數(shù)內、類型定義內等)定義宏,除非它只在該局部范圍內使用,否則會損害程序的清晰性。
3 宏的常見用法
- 防止一個頭文件被重復包含
#ifndef COMDEF_H
#define COMDEF_H
//頭文件內容
#endif
- 得到指定地址上的一個字節(jié)或字
#define MEM_B(x) (*((byte *)(x)))
#define MEM_W(x) (*((word *)(x)))
- 求最大值和最小值
#define MAX(x,y) (((x)>(y)) ? (x) : (y))
#define MIN(x,y) (((x) < (y)) ? (x) : (y))
- 得到一個field在結構體(struct)中的偏移量
#define FPOS(type,field) ((dword)