寫好C語言,漂亮的宏定義很重要!
來源:玩轉(zhuǎn)嵌入式
寫好C語言,漂亮的宏定義很重要!使用宏定義可以防止出錯,提高可移植性,可讀性,方便性等。下面列舉了一些成熟軟件中常用的宏定義。
來源:玩轉(zhuǎn)嵌入式
1. 防止一個頭文件被重復(fù)包含
1#ifndef?COMDEF_H
2#define?COMDEF_H
3//頭文件內(nèi)容
4#endif
1#ifndef?COMDEF_H
2#define?COMDEF_H
3//頭文件內(nèi)容
4#endif
2. 重新定義一些類型,防止由于各種平臺和編譯器的不同,而產(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?*/
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. 得到指定地址上的一個字節(jié)或字
1#define?MEM_B(?x?)?(?*(?(byte?*)?(x)?)?)
2#define?MEM_W(?x?)?(?*(?(word?*)?(x)?)?)
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)?)
1#define?MAX(?x,?y?)?(?((x)?>?(y))???(x)?:?(y)?)
2#define?MIN(?x,?y?)?(?((x)?(y))???(x)?:?(y)?)
5. 得到一個field在結(jié)構(gòu)體(struct)中的偏移量
1#define?FPOS(?type,?field?)?\
2/*lint?-e545?*/?(?(dword)?&((?type?*)?0)->?field?)?/*lint?+e545?*/
1#define?FPOS(?type,?field?)?\
2/*lint?-e545?*/?(?(dword)?&((?type?*)?0)->?field?)?/*lint?+e545?*/
6. 得到一個結(jié)構(gòu)體中field所占用的字節(jié)數(shù)
1#define?FSIZ(?type,?field?)?sizeof(?((type?*)?0)->field?)
1#define?FSIZ(?type,?field?)?sizeof(?((type?*)?0)->field?)
7. 按照LSB格式把兩個字節(jié)轉(zhuǎn)化為一個Word
1#define?FLIPW(?ray?)?(?(((word)?(ray)[0])?*?256)?+?(ray)[1]?)
1#define?FLIPW(?ray?)?(?(((word)?(ray)[0])?*?256)?+?(ray)[1]?)
8. 按照LSB格式把一個Word轉(zhuǎn)化為兩個字節(jié)
1#define?FLOPW(?ray,?val?)?\
2(ray)[0]?=?((val)?/?256);?\
3(ray)[1]?=?((val)?&?0xFF)
1#define?FLOPW(?ray,?val?)?\
2(ray)[0]?=?((val)?/?256);?\
3(ray)[1]?=?((val)?&?0xFF)
9. 得到一個變量的地址(word寬度)
1#define?B_PTR(?var?)?(?(byte?*)?(void?*)?&(var)?)
2#define?W_PTR(?var?)?(?(word?*)?(void?*)?&(var)?)
1#define?B_PTR(?var?)?(?(byte?*)?(void?*)?&(var)?)
2#define?W_PTR(?var?)?(?(word?*)?(void?*)?&(var)?)
10. 得到一個字的高位和低位字節(jié)
1#define?WORD_LO(xxx)?((byte)?((word)(xxx)?&?255))
2#define?WORD_HI(xxx)?((byte)?((word)(xxx)?>>?8))
1#define?WORD_LO(xxx)?((byte)?((word)(xxx)?&?255))
2#define?WORD_HI(xxx)?((byte)?((word)(xxx)?>>?8))
11. 返回一個比X大的最接近的8的倍數(shù)
1#define?RND8(?x?)?((((x)?+?7)?/?8?)?*?8?)
1#define?RND8(?x?)?((((x)?+?7)?/?8?)?*?8?)
12. 將一個字母轉(zhuǎn)換為大寫
1#define?UPCASE(?c?)?(?((c)?>=?'a'?&&?(c)?<=?'z')???((c)?-?0x20)?:?(c)?)
1#define?UPCASE(?c?)?(?((c)?>=?'a'?&&?(c)?<=?'z')???((c)?-?0x20)?:?(c)?)
13. 判斷字符是不是10進制的數(shù)字
1#define?DECCHK(?c?)?((c)?>=?'0'?&&?(c)?<=?'9')
1#define?DECCHK(?c?)?((c)?>=?'0'?&&?(c)?<=?'9')
14. 判斷字符是不是16進制的數(shù)字
1#define?HEXCHK(?c?)?(?((c)?>=?'0'?&&?(c)?<=?'9')?||\
2((c)?>=?'A'?&&?(c)?<=?'F')?||\
3((c)?>=?'a'?&&?(c)?<=?'f')?)
1#define?HEXCHK(?c?)?(?((c)?>=?'0'?&&?(c)?<=?'9')?||\
2((c)?>=?'A'?&&?(c)?<=?'F')?||\
3((c)?>=?'a'?&&?(c)?<=?'f')?)
15. 防止溢出的一個方法
1#define?INC_SAT(?val?)?(val?=?((val)+1?>?(val))???(val)+1?:?(val))
1#define?INC_SAT(?val?)?(val?=?((val)+1?>?(val))???(val)+1?:?(val))
16. 返回數(shù)組元素的個數(shù)
1#define?ARR_SIZE(?a?)?(?sizeof(?(a)?)?/?sizeof(?(a[0])?)?)
1#define?ARR_SIZE(?a?)?(?sizeof(?(a)?)?/?sizeof(?(a[0])?)?)
17. 返回一個無符號數(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)?)
1#define?MOD_BY_POWER_OF_TWO(?val,?mod_by?)?\
2(?(dword)(val)?&?(dword)((mod_by)-1)?)
18. 對于IO空間映射在存儲空間的結(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)))
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標準說明了五個預(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?_
如果編譯不是標準的,則可能僅支持以上宏名中的幾個,或根本不支持。記住編譯程序也許還提供其它預(yù)定義的宏名。 _ L I N E _及_ F I L E _宏指令在有關(guān)# l i n e的部分中已討論,這里討論其余的宏名。_ D AT E _宏指令含有形式為月/日/年的串,表示源文件被翻譯到代碼時的日期。源代碼翻譯到目標代碼的時間作為串包含在_ T I M E _中。串形式為時:分:秒。 如果實現(xiàn)是標準的,則宏_ S T D C _含有十進制常量1。如果它含有任何其它數(shù),則實現(xiàn)是非標準的。 可以定義宏,例如:當定義了_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
A N S I標準說明了五個預(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?_
_ L I N E _及_ F I L E _宏指令在有關(guān)# l i n e的部分中已討論,這里討論其余的宏名。_ D AT E _宏指令含有形式為月/日/年的串,表示源文件被翻譯到代碼時的日期。_ T I M E _中。串形式為時:分:秒。_ S T D C _含有十進制常量1。如果它含有任何其它數(shù),則實現(xiàn)是非標準的。_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. 宏定義防止使用時錯誤用小括號包含。
例如:
1#define?ADD(a,b)?(a+b)
用do{}while(0)語句包含多語句防止錯誤,例如:
1#difne?DO(a,b)?a+b;\
2a++;
應(yīng)用時:
1if(….)
2DO(a,b);?//產(chǎn)生錯誤
3else
解決方法:
1#define?DO(a,b)?do{a+b;\
2a++;}while(0)
例如:
1#define?ADD(a,b)?(a+b)
用do{}while(0)語句包含多語句防止錯誤,例如:
1#difne?DO(a,b)?a+b;\
2a++;
應(yīng)用時:
1if(….)
2DO(a,b);?//產(chǎn)生錯誤
3else
解決方法:
1#define?DO(a,b)?do{a+b;\
2a++;}while(0)
