天天看點

#define宏的進階用法

參考:https://blog.csdn.net/xiahouzuoxin/article/details/9494503

一 、宏的定義與撤銷

#普通宏定義
#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]) ) )
           

繼續閱讀