#define用法集錦

1.?簡單的define定義

#define MAXTIME 1000

一個簡單的MAXTIME就定義好了,它代表1000,如果在程序里面寫

if(i<MAXTIME){.........}

編譯器在處理這個代碼之前會對MAXTIME進行處理替換為1000。

這樣的定義看起來類似于普通的常量定義CONST,但也有著不同,因為define的定義更像是簡單的文本替換,而不是作為一個量來使用,這個問題在下面反映的尤為突出。

2.define函數定義

define可以像函數那樣接受一些參數,如下

#define max(x,y) (x)>(y)?(x):(y);

這個定義就將返回兩個數中較大的那個,看到了嗎?因為這個“函數”沒有類型檢查,就好像一個函數模板似的,當然,它絕對沒有模板那么安全就是了??梢宰鳛橐粋€簡單的模板來使用而已。

但是這樣做的話存在隱患,例子如下:

#define Add(a,b) a+b;

在一般使用的時候是沒有問題的,但是如果遇到如:c * Add(a,b) * d的時候就會出現問題,代數式的本意是a+b然后去和c,d相乘,但是因為使用了define(它只是一個簡單的替換),所以式子實際上變成了

c*a + b*d

另外舉一個例子:

#define pin (int*);

pin a,b;

本意是a和b都是int型指針,但是實際上變成int* a,b;

a是int型指針,而b是int型變量。

這是應該使用typedef來代替define,這樣a和b就都是int型指針了。

所以我們在定義的時候,養(yǎng)成一個良好的習慣,建議所有的層次都要加括號。

3.宏的單行定義(少見用法)

#define A(x) T_##x

#define B(x) #@x

#define C(x) #x

我們假設:x=1,則有:

A(1)------〉T_1

B(1)------〉'1'

C(1)------〉"1"

(這里參考了 hustli的文章)

3.define的多行定義

define可以替代多行的代碼,例如MFC中的宏定義(非常的經典,雖然讓人看了惡心)

#define MACRO(arg1, arg2) do { \

/* declarations */ \

stmt1; \

stmt2; \

/* ... */ \

} while(0) /* (no trailing ; ) */

關鍵是要在每一個換行的時候加上一個"\"

4.在大規(guī)模的開發(fā)過程中,特別是跨平臺和系統(tǒng)的軟件里,define最重要的功能是條件編譯。

就是:

#ifdef WINDOWS

......

......

#endif

#ifdef LINUX

......

......

#endif

可以在編譯的時候通過#define設置編譯環(huán)境

5.如何定義宏、取消宏

//定義宏

#define [MacroName] [MacroValue]

//取消宏

#undef [MacroName]

//普通宏

#define PI (3.1415926)

帶參數的宏

#define max(a,b) ((a)>(b)? (a),(b))

關鍵是十分容易產生錯誤,包括機器和人理解上的差異等等。

6.條件編譯

#ifdef XXX…(#else) … #endif

例如

#ifdef DV22_AUX_INPUT

#define AUX_MODE 3?

#else

#define AUY_MODE 3

#endif

#ifndef XXX … (#else) … #endif

7.頭文件(.h)可以被頭文件或C文件包含;

重復包含(重復定義)

由于頭文件包含可以嵌套,那么C文件就有可能包含多次同一個頭文件,就可能出現重復定義的問題的。

通過條件編譯開關來避免重復包含(重復定義)

例如

#ifndef __headerfileXXX__

#define __headerfileXXX__

//文件內容

#endif



?Instances


1、防止一個頭文件被重復包含

#ifndef COMDEF_H

#define COMDEF_H

?//頭文件內容

#endif

2、重新定義一些類型,防止由于各種平臺和編譯器的不同,而產生的類型字節(jié)數差異,方便移植。

typedef??unsigned char??????boolean;???? /* Boolean value type. */

typedef??unsigned long int??uint32;??????/* Unsigned 32 bit value */

typedef??unsigned short???? uint16;??????/* Unsigned 16 bit value */

typedef??unsigned char??????uint8;?????? /* Unsigned 8??bit value */

typedef??signed long int????int32;?????? /* Signed 32 bit value */

typedef??signed short?????? int16;?????? /* Signed 16 bit value */

typedef??signed char????????int8;????????/* Signed 8??bit value */

//下面的不建議使用

typedef??unsigned char???? byte;???????? /* Unsigned 8??bit value type. */

typedef??unsigned short????word;???????? /* Unsinged 16 bit value type. */

typedef??unsigned long???? dword;????????/* Unsigned 32 bit value type. */

typedef??unsigned char???? uint1;????????/* Unsigned 8??bit value type. */

typedef??unsigned short????uint2;????????/* Unsigned 16 bit value type. */

typedef??unsigned long???? uint4;????????/* Unsigned 32 bit value type. */

typedef??signed char?????? int1;???????? /* Signed 8??bit value type. */

typedef??signed short??????int2;???????? /* Signed 16 bit value type. */

typedef??long int??????????int4;???????? /* Signed 32 bit value type. */

typedef??signed long?????? sint31;?????? /* Signed 32 bit value */

typedef??signed short??????sint15;?????? /* Signed 16 bit value */

typedef??signed char?????? sint7;????????/* Signed 8??bit value */

3、得到指定地址上的一個字節(jié)或字

#define??MEM_B( x )??( *( (byte *) (x) ) )

#define??MEM_W( x )??( *( (word *) (x) ) )

4、求最大值和最小值

?? #define??MAX( x, y ) ( ((x) > (y)) ? (x) : (y) )

?? #define??MIN( x, y ) ( ((x) < (y)) ? (x) : (y) )

5、得到一個field在結構體(struct)中的偏移量

#define FPOS( type, field ) \

/*lint -e545 */ ( (dword) &(( type *) 0)-> field ) /*lint +e545 */

6、得到一個結構體中field所占用的字節(jié)數

#define FSIZ( type, field ) sizeof( ((type *) 0)->field )

7、按照LSB格式把兩個字節(jié)轉化為一個Word

#define??FLIPW( ray ) ( (((word) (ray)[0]) * 256) + (ray)[1] )

8、按照LSB格式把一個Word轉化為兩個字節(jié)

#define??FLOPW( ray, val ) \

??(ray)[0] = ((val) / 256); \

??(ray)[1] = ((val) & 0xFF)

9、得到一個變量的地址(word寬度)

#define??B_PTR( var )??( (byte *) (void *) &(var) )

#define??W_PTR( var )??( (word *) (void *) &(var) )

10、得到一個字的高位和低位字節(jié)

#define??WORD_LO(xxx)??((byte) ((word)(xxx) & 255))

#define??WORD_HI(xxx)??((byte) ((word)(xxx) >> 8))

11、返回一個比X大的最接近的8的倍數

#define RND8( x )?????? ((((x) + 7) / 8 ) * 8 )

12、將一個字母轉換為大寫

#define??UPCASE( c ) ( ((c) >= 'a' && (c) <= 'z') ? ((c) - 0x20) : (c) )

13、判斷字符是不是10進值的數字

#define??DECCHK( c ) ((c) >= '0' && (c) <= '9')

14、判斷字符是不是16進值的數字

#define??HEXCHK( c ) ( ((c) >= '0' && (c) <= '9') ||\

?????????????????????? ((c) >= 'A' && (c) <= 'F') ||\

((c) >= 'a' && (c) <= 'f') )

15、防止溢出的一個方法

#define??INC_SAT( val )??(val = ((val)+1 > (val)) ? (val)+1 : (val))

16、返回數組元素的個數

#define??ARR_SIZE( a )??( sizeof( (a) ) / sizeof( (a[0]) ) )

17、返回一個無符號數n尾的值MOD_BY_POWER_OF_TWO(X,n)=X%(2^n)

#define MOD_BY_POWER_OF_TWO( val, mod_by ) \

?????????? ( (dword)(val) & (dword)((mod_by)-1) )

18、對于IO空間映射在存儲空間的結構,輸入輸出處理

??#define inp(port)???????? (*((volatile byte *) (port)))

??#define inpw(port)????????(*((volatile word *) (port)))

??#define inpdw(port)?????? (*((volatile dword *)(port)))

??#define outp(port, val)?? (*((volatile byte *) (port)) = ((byte) (val)))

??#define outpw(port, val)??(*((volatile word *) (port)) = ((word) (val)))

??#define outpdw(port, val) (*((volatile dword *) (port)) = ((dword) (val)))

19、使用一些宏跟蹤調試

ANSI標準說明了五個預定義的宏名。它們是:

__LINE__

__FILE__

__DATE__

__TIME__

__STDC__

C++中還定義了 __cplusplus

如果編譯器不是標準的,則可能僅支持以上宏名中的幾個,或根本不支持。記住編譯程序也許還提供其它預定義的宏名。

__LINE__ 及 __FILE__ 宏指示,#line指令可以改變它的值,簡單的講,編譯時,它們包含程序的當前行數和文件名。

__DATE__ 宏指令含有形式為月/日/年的串,表示源文件被翻譯到代碼時的日期。

__TIME__ 宏指令包含程序編譯的時間。時間用字符串表示,其形式為:分:秒

__STDC__ 宏指令的意義是編譯時定義的。一般來講,如果__STDC__已經定義,編譯器將僅接受不包含任何非標準擴展的標準C/C++代碼。如果實現是標準的,則宏__STDC__含有十進制常量1。如果它含有任何其它數,則實現是非標準的。

__cplusplus 與標準c++一致的編譯器把它定義為一個包含至少6為的數值。與標準c++不一致的編譯器將使用具有5位或更少的數值。

可以定義宏,例如:

當定義了_DEBUG,輸出數據信息和所在文件所在行

#ifdef _DEBUG

#define DEBUGMSG(msg,date) printf(msg);printf(“%d%d%d”,date,_LINE_,_FILE_)

#else

#define DEBUGMSG(msg,date)

#endif

20、宏定義防止錯誤使用小括號包含。

例如:

有問題的定義:#define DUMP_WRITE(addr,nr) {memcpy(bufp,addr,nr); bufp += nr;}

應該使用的定義: #difne DO(a,b) do{a+b;a++;}while(0)

例如:

if(addr)

??? DUMP_WRITE(addr,nr);

else

??? do_somethong_else();

宏展開以后變成這樣:

if(addr)

??? {memcpy(bufp,addr,nr); bufp += nr;};

else

??? do_something_else();

gcc在碰到else前面的“;”時就認為if語句已經結束,因而后面的else不在if語句中。而采用do{} while(0)的定義,在任何情況下都沒有問題。而改為 #difne DO(a,b) do{a+b;a++;}while(0) 的定義則在任何情況下都不會出錯

?

21.?define中的特殊標識符

#define Conn(x,y) x##y

#define ToChar(x) #@x

#define ToString(x) #x

int a=Conn(12,34);

char b=ToChar(a);

char c[]=ToString(a);

結果是 a=1234,c='a',c='1234';

可以看出 ## 是簡單的連接符,#@用來給參數加單引號,#用來給參數加雙引號即轉成字符串

免費C/C++基礎丶進階資料,還有實踐課程免費領,加群728483370

?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
【社區(qū)內容提示】社區(qū)部分內容疑似由AI輔助生成,瀏覽時請結合常識與多方信息審慎甄別。
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發(fā)布,文章內容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務。

相關閱讀更多精彩內容

  • Lua 5.1 參考手冊 by Roberto Ierusalimschy, Luiz Henrique de F...
    蘇黎九歌閱讀 14,246評論 0 38
  • 在C語言中,五種基本數據類型存儲空間長度的排列順序是: A)char B)char=int<=float C)ch...
    夏天再來閱讀 4,039評論 0 2
  • 姓名:呂彬 學號:1613014035 【嵌牛導讀】#define 宏定義是個演技非常高超的替身演員,但也會經常耍...
    傻彬兒閱讀 1,359評論 0 1
  • pyspark.sql模塊 模塊上下文 Spark SQL和DataFrames的重要類: pyspark.sql...
    mpro閱讀 9,916評論 0 13
  • - 1 - 技術方向 1. 測試開發(fā)工程師: 門檻:必須具備非常強的代碼能力,最好有一點測試思維,能理解測試部門的...
    CuteProgrammer閱讀 329評論 0 3

友情鏈接更多精彩內容