一个复杂的宏定义代码如何看懂是个头疼的事儿,
本文介绍将一个复杂的宏定义代码打印成标准代码格式
宏定义输出
下面举例一个复杂的宏定义
#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
本文深入探讨了如何将复杂宏定义代码转换为易于理解的标准代码形式,并通过实例展示了具体操作过程。包括宏定义输出、C/C++宏定义中#与##的区别及其应用,以及详细解析复杂宏定义的步骤。

被折叠的 条评论
为什么被折叠?



