?
寫好C語言,漂亮的宏定義很重要,使用宏定義可以防止出錯(cuò),提高可移植性,可讀性,方便性等等。下面列舉一些成熟軟件中常用的宏定義。
1. 防止一個(gè)頭文件被重復(fù)包含
?
1#ifndef?COMDEF_H 2#define?COMDEF_H 3//頭文件內(nèi)容 4#endif
2. 重新定義一些類型,防止由于各種平臺(tái)和編譯器的不同,而產(chǎn)生的類型字節(jié)數(shù)差異,方便移植。
?
1typedef?unsigned?char?boolean;?/*?Boolean?value?type.?*/ 2typedef?unsigned?long?int?uint32;?/*?Unsigned?32?bit?value?*/ 3typedef?unsigned?short?uint16;?/*?Unsigned?16?bit?value?*/ 4typedef?unsigned?char?uint8;?/*?Unsigned?8?bit?value?*/ 5typedef?signed?long?int?int32;?/*?Signed?32?bit?value?*/ 6typedef?signed?short?int16;?/*?Signed?16?bit?value?*/ 7typedef?signed?char?int8;?/*?Signed?8?bit?value?*/
下面的不建議使用
1typedef?unsigned?char?byte;?/*?Unsigned?8?bit?value?type.?*/ 2typedef?unsigned?short?word;?/*?Unsinged?16?bit?value?type.?*/ 3typedef?unsigned?long?dword;?/*?Unsigned?32?bit?value?type.?*/ 4typedef?unsigned?char?uint1;?/*?Unsigned?8?bit?value?type.?*/ 5typedef?unsigned?short?uint2;?/*?Unsigned?16?bit?value?type.?*/ 6typedef?unsigned?long?uint4;?/*?Unsigned?32?bit?value?type.?*/ 7typedef?signed?char?int1;?/*?Signed?8?bit?value?type.?*/ 8typedef?signed?short?int2;?/*?Signed?16?bit?value?type.?*/ 9typedef?long?int?int4;?/*?Signed?32?bit?value?type.?*/ 10typedef?signed?long?sint31;?/*?Signed?32?bit?value?*/ 11typedef?signed?short?sint15;?/*?Signed?16?bit?value?*/ 12typedef?signed?char?sint7;?/*?Signed?8?bit?value?*/
3. 得到指定地址上的一個(gè)字節(jié)或字
?
1#define?MEM_B(?x?)?(?*(?(byte?*)?(x)?)?) 2#define?MEM_W(?x?)?(?*(?(word?*)?(x)?)?)
4. 求最大值和最小值
?
1#define?MAX(?x,?y?)?(?((x)?>?(y))???(x)?:?(y)?) 2#define?MIN(?x,?y?)?(?((x)?(y))???(x)?:?(y)?)
5. 得到一個(gè)field在結(jié)構(gòu)體(struct)中的偏移量
?
1#define?FPOS(?type,?field?)? 2/*lint?-e545?*/?(?(dword)?&((?type?*)?0)->?field?)?/*lint?+e545?*/
6. 得到一個(gè)結(jié)構(gòu)體中field所占用的字節(jié)數(shù)
?
1#define?FSIZ(?type,?field?)?sizeof(?((type?*)?0)->field?)
7. 按照LSB格式把兩個(gè)字節(jié)轉(zhuǎn)化為一個(gè)Word
?
1#define?FLIPW(?ray?)?(?(((word)?(ray)[0])?*?256)?+?(ray)[1]?)
8. 按照LSB格式把一個(gè)Word轉(zhuǎn)化為兩個(gè)字節(jié)
?
1#define?FLOPW(?ray,?val?)? 2(ray)[0]?=?((val)?/?256);? 3(ray)[1]?=?((val)?&?0xFF)
9. 得到一個(gè)變量的地址(word寬度)
?
1#define?B_PTR(?var?)?(?(byte?*)?(void?*)?&(var)?) 2#define?W_PTR(?var?)?(?(word?*)?(void?*)?&(var)?)
10. 得到一個(gè)字的高位和低位字節(jié)
?
1#define?WORD_LO(xxx)?((byte)?((word)(xxx)?&?255)) 2#define?WORD_HI(xxx)?((byte)?((word)(xxx)?>>?8))
11. 返回一個(gè)比X大的最接近的8的倍數(shù)
?
1#define?RND8(?x?)?((((x)?+?7)?/?8?)?*?8?)
12. 將一個(gè)字母轉(zhuǎn)換為大寫
?
1#define?UPCASE(?c?)?(?((c)?>=?'a'?&&?(c)?<=?'z')???((c)?-?0x20)?:?(c)?)
13. 判斷字符是不是10進(jìn)制的數(shù)字
?
1#define?DECCHK(?c?)?((c)?>=?'0'?&&?(c)?<=?'9')
14. 判斷字符是不是16進(jìn)制的數(shù)字
?
1#define?HEXCHK(?c?)?(?((c)?>=?'0'?&&?(c)?<=?'9')?|| 2((c)?>=?'A'?&&?(c)?<=?'F')?|| 3((c)?>=?'a'?&&?(c)?<=?'f')?)
15. 防止溢出的一個(gè)方法
?
1#define?INC_SAT(?val?)?(val?=?((val)+1?>?(val))???(val)+1?:?(val))
16. 返回?cái)?shù)組元素的個(gè)數(shù)
?
1#define?ARR_SIZE(?a?)?(?sizeof(?(a)?)?/?sizeof(?(a[0])?)?)
17. 返回一個(gè)無符號(hào)數(shù)n尾的值MOD_BY_POWER_OF_TWO(X,n)=X%(2^n)
?
1#define?MOD_BY_POWER_OF_TWO(?val,?mod_by?)? 2(?(dword)(val)?&?(dword)((mod_by)-1)?)
18. 對(duì)于IO空間映射在存儲(chǔ)空間的結(jié)構(gòu),輸入輸出處理
?
1#define?inp(port)?(*((volatile?byte?*)?(port))) 2#define?inpw(port)?(*((volatile?word?*)?(port))) 3#define?inpdw(port)?(*((volatile?dword?*)(port))) 4#define?outp(port,?val)?(*((volatile?byte?*)?(port))?=?((byte)?(val))) 5#define?outpw(port,?val)?(*((volatile?word?*)?(port))?=?((word)?(val))) 6#define?outpdw(port,?val)?(*((volatile?dword?*)?(port))?=?((dword)?(val)))
19. 使用一些宏跟蹤調(diào)試
?
A N S I標(biāo)準(zhǔn)說明了五個(gè)預(yù)定義的宏名。它們是:
1_?L?I?N?E?_ 2_?F?I?L?E?_ 3_?D?A?T?E?_ 4_?T?I?M?E?_ 5_?S?T?D?C?_
如果編譯不是標(biāo)準(zhǔn)的,則可能僅支持以上宏名中的幾個(gè),或根本不支持。記住編譯程序也許還提供其它預(yù)定義的宏名。
_ L I N E _
及_ F I L E _
宏指令在有關(guān)# l i n e
的部分中已討論,這里討論其余的宏名。
_ D AT E _
宏指令含有形式為月/日/年的串,表示源文件被翻譯到代碼時(shí)的日期。
源代碼翻譯到目標(biāo)代碼的時(shí)間作為串包含在_ T I M E _
中。串形式為時(shí):分:秒。
如果實(shí)現(xiàn)是標(biāo)準(zhǔn)的,則宏_ S T D C _
含有十進(jìn)制常量1。如果它含有任何其它數(shù),則實(shí)現(xiàn)是非標(biāo)準(zhǔn)的。
可以定義宏,例如: 當(dāng)定義了_DEBUG
,輸出數(shù)據(jù)信息和所在文件所在行
1#ifdef?_DEBUG 2#define?DEBUGMSG(msg,date)?printf(msg);printf(“%d%d%d”,date,_LINE_,_FILE_) 3#else 4#define?DEBUGMSG(msg,date) 5#endif
20. 宏定義防止使用時(shí)錯(cuò)誤用小括號(hào)包含。
?
例如:
1#define?ADD(a,b)?(a+b)
用do{}while(0)
語句包含多語句防止錯(cuò)誤
例如:
1#difne?DO(a,b)?a+b; 2a++;
應(yīng)用時(shí):
1if(….) 2DO(a,b);?//產(chǎn)生錯(cuò)誤 3else
解決方法:
1#difne?DO(a,b)?do{a+b; 2a++;}while(0)
? 審核編輯:湯梓紅
評(píng)論
查看更多