===========================================================
define中的三個特殊符號:#,##,#@
===========================================================
#define Conn(x,y) x##y
#define ToChar(x) #@x
#define ToString(x) #x
(1)x##y表示什麼?表示x連接y,舉例說:
int n = Conn(123,456); /* 結果就是n=123456;*/
char* str = Conn("asdf", "adf"); /*結果就是 str = "asdfadf";*/
(2)再來看#@x,其實就是給x加上單引號,結果返回是一個const char。舉例說:
char a = ToChar(1);結果就是a='1';
做個越界試驗char a = ToChar(123);結果就錯了;
但是如果你的參數超過四個字符,編譯器就給給你報錯了!
error C2015: too many characters in constant :P
(3)最後看看#x,估計你也明白了,他是給x加雙引號
char* str = ToString(123132);就成了str="123132";
===========================================================
常用的一些宏定義
===========================================================
1 防止一個頭文件被重復包含
#ifndef BODYDEF_H
#define BODYDEF_H
//頭文件內容
#endif
2 得到指定地址上的一個字節或字
#define MEM_B( x ) ( *( (byte *) (x) ) )
#define MEM_W( x ) ( *( (word *) (x) ) )
用法如下:
#include <iostream>
#include <windows.h>
#define MEM_B(x) (*((byte*)(x)))
#define MEM_W(x) (*((WORD*)(x)))
int main()
{
int bTest = 0x123456;
byte m = MEM_B((&bTest));/*m=0x56*/
int n = MEM_W((&bTest));/*n=0x3456*/
return 0;
}
3 得到一個field在結構體(struct)中的偏移量
#define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )
4 得到一個結構體中field所占用的字節數
#define FSIZ( type, field ) sizeof( ((type *) 0)->field )
5 得到一個變量的地址(word寬度)
#define B_PTR( var ) ( (byte *) (void *) &(var) )
#define W_PTR( var ) ( (word *) (void *) &(var) )
6 將一個字母轉換為大寫
#define UPCASE( c ) ( ((c) >= ''a'' && (c) <= ''z'') ? ((c) - 0x20) : (c) )
7 判斷字符是不是10進值的數字
#define DECCHK( c ) ((c) >= ''0'' && (c) <= ''9'')
8 判斷字符是不是16進值的數字
#define HEXCHK( c ) ( ((c) >= ''0'' && (c) <= ''9'') ||((c) >= ''A'' && (c) <= ''F'') ||((c) >= ''a'' && (c) <= ''f'') )
9 防止溢出的一個方法
#define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))
10 返回數組元素的個數
#define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )
11 使用一些宏跟蹤調試
在調試時,我們可以設置__DEBUG宏,也可以再Makefile中使用-D編譯選項設置,
[cpp]
#define __DEBUG
#define __DEBUG使用方法為,[cpp] view plaincopyprint?
#ifdef __DEBUG
printf("%s", ...);
#endif
#ifdef __DEBUG
printf("%s", ...);
#endif另外,ANSI C標准中有幾個標准預定義宏,前面幾個常用於printf(sprintf)等語句中:
__LINE__:在源代碼中插入當前源代碼行號;
__FILE__:在源文件中插入當前源文件名;
__DATE__:在源文件中插入當前的編譯日期
__TIME__:在源文件中插入當前編譯時間;
__STDC__:當要求程序嚴格遵循ANSI C標准時該標識被賦值為1;
__cplusplus:當編寫C++程序時該標識符被定義。
其中__cplusplus常用於頭文件中,格式如下:
[cpp]
#ifndef _ZX_FUNC_H
#define _ZX_FUNC_H
#ifdef __cplusplus
extern "C" {
#endif
/* functions */
char *strdup (const char *s);
#ifdef __cplusplus
}
#endif
#endif
#ifndef _ZX_FUNC_H
#define _ZX_FUNC_H
#ifdef __cplusplus
extern "C" {
#endif
/* functions */
char *strdup (const char *s);
#ifdef __cplusplus
}
#endif
#endif
extern"C"表示將其中的代碼按照C編譯方法編譯,目的是實現C++與C語言的調用。
C編譯與C++編譯的區別是:C會將上面strdup編譯成_STRDUP符號,而C++會編譯成_STRDUP_CHAR,這也是C++為什麼能實現函數重載的原因。extern只能出現在C++文件中,一般如上面的方式置於頭文件中。
要在C中調用C++代碼,需要在C代碼中的函數或變量聲明為extern類型,在C++中將函數或變量用extern "C"修飾。
12 簡單數學計算(絕對值,三角函數等)
[cpp]
#define ABS( a ) ( ((a)>0) ? (a) : (-(a)) )
#define ABS( a ) ( ((a)>0) ? (a) : (-(a)) )
13 #define 一個復雜語句
比如交換a,b的值,
[cpp]
#define(a,b) do { \
int t = 0;
t = a; \
a = b; \
b = t; \
} while(0)
#define(a,b) do { \
int t = 0;
t = a; \
a = b; \
b = t; \
} while(0)
注:#define的這些高級用法在Linux源代碼很多處出現,可閱讀參考Linux源代碼。