1. 程式人生 > >C/C++常用巨集定義

C/C++常用巨集定義

#ifndef COMDEF_H 
#define COMDEF_H 
//標頭檔案內容 ...
#endif 

2,重新定義一些型別,防止由於各種平臺和編譯器的不同,而產生的型別位元組數差異,方便移植。 
typedef  unsigned long int  uint32;      /* Unsigned 32 bit value */ 

3,得到指定地址上的一個位元組或字 
#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 )   ( (dword) &(( type *) 0)-> field )

6,得到一個結構體中field所佔用的位元組數 
#define FSIZ( type, field ) sizeof( ((type *) 0)->field ) 

7,按照LSB格式把兩個位元組轉化為一個word 
#define  FLIPW( ray ) ( (((word) (ray)[0]) * 256) + (ray)[1] ) 

8,按照LSB格式把一個word轉化為兩個位元組 
#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,得到一個字的高位和低位位元組 
#define  WORD_LO(xxx)  ((byte) ((word)(var) & 255))
#define  WORD_HI(xxx)  ((byte) ((word)(var) >> 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,對於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))) 

18,使用一些巨集跟蹤除錯 
ANSI標準說明了五個預定義的巨集名。它們是: 
__LINE__
__FILE__
__DATE__
__TIME__ 
__STDC__

如果編譯不是標準的,則可能僅支援以上巨集名中的幾個,或根本不支援。記住編譯程式 也許還提供其它預定義的巨集名。 
\ 是行連線符,會將下一行和前一行連線成為一行,即將物理上的兩行連線成邏輯上的一行
__FILE__ 是內建巨集 代表原始檔的檔名 
__LINE__ 是內建巨集,代表該行程式碼的所在行號
__DATE__巨集指令含有形式為月/日/年的串,表示原始檔被翻譯到程式碼時的日期。 
原始碼翻譯到目的碼的時間作為串包含在__TIME__ 中。串形式為時:分:秒。 
如果實現是標準的,則巨集__STDC__含有十進位制常量1。如果它含有任何其它數,則實現是非標準的。 

可以定義巨集,例如: 
當定義了_DEBUG,輸出資料資訊和所在檔案所在行 

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

19,巨集定義防止使用是錯誤 
用小括號包含。 
例如:#define ADD(a,b) (a+b)