一,宏的定義與撤銷
#普通宏定義 #define PI 3.14 //編譯階段替換掉宏 #define T1 3+4 //容易產生歧義 #define T2 (3+4) //添加括號后,語義清楚 float r = 1.0; float area = PI * r * r; int a = 2* T1 #宏替換后變成 int a = 2*3+4 不符合本意 ing a = 2* T2 #紅替換后變成 int a = 2*(3+4) 符合本意 #undef PI float area = PI * r * r; #error: ‘PI’ was not declared in this scope //引號中的宏定義不會被替換 printf("%s:%f\n", "PI", PI); //輸出 PI:3.14 //宏定義的名字必須是合法標識符 #define 0x abcd //error 不能以數字開始 //宏定義中雙引號和單引號必須成對出現 #define TEST11 "Z //error #define TEST2 'Z //error
二、帶有參數的宏定義
//max和min的宏定義帶參數 #define MAX(a,b) (a>b ? a:b) #define MIN(a,b) (a<b ? a:b) //使用帶參數的宏定義 int sum= MAX(1,2) + MIN(1,2); //替換后語句為:int sum = (1>2 ? 1:2) + (1<2 ? 1:2) //參數個數必須宏定義時形參的個數相同 MAX(1,2,3); //會報錯 #undef MAX //撤銷MAX的宏定義 MAX(1,2); //error: ‘MAX’ was not declared in this scope
三、跨行的宏定義 使用反引號\連接
#定義一個交換數值的多行宏,使用反斜杠連接不同行 #define SWAP(a,b) do { \ int t = 0;\ t = a; \ a = b; \ b = t; \ } while(0)
四,三個特殊符號:#,##,#@
#define CONNECT(a,b) a##b #define TOCHAR(a) #@a #define TOSTRING(a) #a //a##b表示連接 int n = CONNECT(123, 456); //結果 n = 123456 char *str = CONNECT("abcd", "efg"); //結果 str = "abcdefg" //@#a 表示用單引號包括參數a,返回的是一個字符 char * ch1 = TOCHAR(1); //結果 ch = '1' char * ch2 = TOCHAR(123); //報錯,單引號只用在單個字符里 //#a 表示用雙引號包括參數a,返回一個字符串 char * str1 = TOSTRING(123); // str = "123"
五、常見的宏定義
- 防止頭文件被重復包含
#ifndef BODYDEF_H #define BODYDEF_H //頭文件內容 #endif
- 得到指定地址上的一個字節值或字值
#include "stdio.h" //B表示字節byte #define MEM_B( x ) ( *( (byte *) (x) ) ) //B表示字word,可以理解為int #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; }
- 得到一個field在結構體(struct)中的偏移量
#define OFFSETOF( type, field ) ( (size_t) &(( type *) 0)-> field )
- 得到一個結構體中field所占用的字節數
#define FSIZ( type, field ) sizeof( ((type *) 0)->field )
- 得到一個變量的地址(word寬度)
#define B_PTR( var ) ( (byte *) (void *) &(var) ) #define W_PTR( var ) ( (word *) (void *) &(var) )
- 將一個字母轉換為大寫
#define UPCASE( c ) ( ((c) >= 'a' && (c) <= 'z') ? ((c) - 0x20) : (c) )
- 判斷字符是不是10進值的數字
#define DECCHK( c ) ((c) >= '0' && (c) <= '9')
- 判斷字符是不是16進值的數字
#define HEXCHK( c ) ( ((c) >= '0' && (c) <= '9') ||((c) >= 'A' && (c) <= 'F') ||((c) >= 'a' && (c) <= 'f') )
- 防止溢出的一個方法
#define INC_SAT( val ) (val = ((val)+1 > (val)) ? (val)+1 : (val))
- 返回數組元素的個數
#define ARR_SIZE( a ) ( sizeof( (a) ) / sizeof( (a[0]) ) )