來(lái),看看這20個(gè)常用的宏定義!
ID:技術(shù)讓夢(mèng)想更偉大
作者:李肖遙
寫好C語(yǔ)言,漂亮的宏定義很重要,使用宏定義可以防止出錯(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è)無(wú)符號(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)說(shuō)明了五個(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)語(yǔ)句包含多語(yǔ)句防止錯(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)
免責(zé)聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺(tái)僅提供信息存儲(chǔ)服務(wù)。文章僅代表作者個(gè)人觀點(diǎn),不代表本平臺(tái)立場(chǎng),如有問(wèn)題,請(qǐng)聯(lián)系我們,謝謝!