一个复杂的宏定义代码如何看懂是个头疼的事儿,
本文介绍将一个复杂的宏定义代码打印成标准代码格式
宏定义输出
下面举例一个复杂的宏定义
#include "stdlib.h"
#include "stdio.h"
#define LD32(a) (*((uint32_t*)(a)))
#define PUT_PIX(OPNAME,OP) \
static void OPNAME ## _pixels8_c(uint8_t *block, const uint8_t *pixels, int line_size, int h){\
int i;\
for(i=0; i<h; i++){\
OP(*((uint32_t*)(block )), LD32(pixels ));\
OP(*((uint32_t*)(block+4)), LD32(pixels+4));\
pixels+=line_size;\
block +=line_size;\
}\
}\
static void OPNAME ## _pixels16_c(uint8_t *block, const uint8_t *pixels, int line_size, int h){\
OPNAME ## _pixels8_c(block , pixels , line_size, h);\
OPNAME ## _pixels8_c(block+8, pixels+8, line_size, h);\
}
#define op_avg(a, b) a = rnd_avg32(a, b)
#define op_put(a, b) a = b
#define _STR(x) #x
#define STR(x) _STR(x)
int main()
{
printf("%s\n",STR(PUT_PIX(put,op_put)));
system("pause");
return 0;
}
C/C++宏定义中#与##区别
## 连接符号由两个井号组成,其功能是在带参数的宏定义中将两个子串(token)联接起来,从而形成一个新的子串。但它不可以是第一个或者最后一个子串。所谓的子串(token)就是指编译器能够识别的最小语法单元。具体的定义在编译原理里有详尽的解释
#符是把传递过来的参数当成字符串进行替代。
假设程序中已经定义了这样一个带参数的宏:
#define PRINT( n ) printf( "token" #n " = %d", token##n )
同时又定义了二个整形变量:
int token9 = 9;
现在在主程序中以下面的方式调用这个宏:
PRINT( 9 );
那么在编译时,上面的这句话被扩展为:
printf( "token" "9" " = %d", token9 );
注意到在这个例子中,PRINT(9);中的这个”9”被原封不动的当成了一个字符串,与”token”连接在了一起,从而成为了token9。而#n也被”9”所替代。
可想而知,上面程序运行的结果就是在屏幕上打印出token9=9
还有点不明白?!
再来一个例子:
#define PRINT( n ) printf( "token" #n " = %d", game##n )
int token9 = 9;
int game9 = 99;
调用:
PRINT(9);
屏幕上打印出:
token9 = 99