C++笔记:宏的高级使用--##、__VA_ARGS__、__FILE__、__FUNCTION__

先说一下本文中会提到的内容:##,__VA_ARGS__, __FILE__, __LINE__ , __FUNCTION__等
宏变量:
先举一个例子,会用到上面这些宏:
[cpp]  view plain copy
  1. #define myprintf(...) printk("[lch]:File:%s, Line:%d, Function:%s," \  
  2.      __VA_ARGS__, __FILE__, __LINE__ ,__FUNCTION__);  

此处的 #define 的作用是将 myprintf( )换成后面那一大串的内容,而括号内 ... 的内容原样抄写在 __VA_ARGS__ 的位置。最终输出如下:

[lch]:File:arch/arm/mach-omap2/board-omap3wscec-camera.c, Line:163, Function:beagle_cam_init,camera init!

解析:

1)__VA_ARGS__:总体来说就是将左边宏中 ... 的内容原样抄写在右边 __VA_ARGS__ 所在的位置。它是一个可变参数的宏,是新的C99规范中新增的,目前似乎只有gcc支持(VC从VC2005开始支持)。要注意的是,printf 的输出格式是括号内左边是字符串,右边是变量,而且右变量与左输出格式是一一对应的。所以在上面那个例子中, __VA_ARGS__只能是一些不含任何变量的字符串常量。因为上面的例子中若__VA_ARGS__含有变量,整个printf的输出与变量便不能一一对应,输出会出错。

如果仅仅是替换函数名,可用如下方式,此时对__VA_ARGS__无任何特殊要求:#define myprintf(...) printk( __VA_ARGS__),在调试程序时可以这样用:

[cpp]  view plain copy
  1. #ifndef LOG_NDEBUG_FUNCTION  
  2. #define LOGFUNC(...) ((void)0)  
  3. #else  
  4. #define LOGFUNC(...) (printk(__VA_ARGS__))  
  5. #endif  
2) __FILE__ :宏在预编译时会替换成当前的源文件名
3) __LINE__:宏在预编译时会替换成当前的行号
4) __FUNCTION__:宏在预编译时会替换成当前的函数名称
5)类似的宏还有 __TIME__,__STDC__, __TIMESTAMP__等,就完全当一个变量来使用即可。

宏连接符##:
举个例子:宏定义为#define XNAME(n) x##n,代码为:XNAME(4),则在预编译时,宏发现XNAME(4)与XNAME(n)匹配,则令 n 为 4,然后将右边的n的内容也变为4,然后将整个XNAME(4)替换为 x##n,亦即 x4,故 最终结果为 XNAME(4) 变为 x4.
代码如下:
[cpp]  view plain copy
  1. #include   
  2. #define XNAME(n) x ## n  
  3. #define PRINT_XN(n) printf("x" #n " = %d/n", x ## n);  
  4. int main(void)  
  5. {  
  6. int XNAME(1) = 14; // becomes int x1 = 14;  
  7. int XNAME(2) = 20; // becomes int x2 = 20;  
  8. PRINT_XN(1); // becomes printf("x1 = %d,", x1);  
  9. PRINT_XN(2); // becomes printf("x2 = %d/n", x2);  
  10. return 0;  
  11. }  
输出为:x1 = 14, x2 = 20


1.#——字符串
假如希望在字符串中包含宏参数,ANSI C允许这样作,在类函数宏的替换部分,#符号用作一个预处理运算符,它可以把语言符号转化程字符串。例如,如果x是一个宏参量,那么#x可以把参数名转化成相应的字符串。该过程称为字符串化(stringizing).
#incldue
#define PSQR(x) printf("the square of" #x "is %d.\n",(x)*(x))
int main(void)
{
    int y =4;
    PSQR(y);
    PSQR(2+4);
    return 0;
}
输出结果:
the square of y is 16.
the square of 2+4 is 36.
第一次调用宏时使用“y”代替#x;第二次调用时用“2+4"代#x。

2.##——连接两个参数
##运算符可以使用类函数宏的替换部分。另外,##还可以用于类对象宏的替换部分。这个运算符把两个语言符号组合成单个语言符号。例如:
#define XNAME(n) x##n
这样宏调用:
XNAME(4)
展开后:
x4
程序:
#include
#define XNAME(x) x##4
#define PXN(n) printf("x"#n" = %d\n",x##n)
int main(void)
{
      int XNAME(1)=12;//int x1=12;
      PXN(1);//printf("x1 = %d\n", x1);
      return 0;
}
 
#include
using namespace std;

#define TEST(pid) (cout< #define TEST2(p) (cout<<#p< int main()
{
    int para3 = 3;
    int para2 = 2;
    TEST(2);    //<==>cout<     TEST(3);    //<==>cout<

    TEST2(test)        //<==>cout<<"test"<     TEST2("test2");    //<==>cout<<""test2""<     system("pause");
    return 0;
}


3.可变宏 ...和_ _VA_ARGS_ _
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,以表示省略号代表什么。比如:
#define PR(...) printf(_ _VA_ARGS_ _)
PR("hello");-->printf("hello");
PR("weight = %d, shipping = $.2f",wt,sp);
      -->printf("weight = %d, shipping = $.2f",wt,sp);
省略号只能代替最后面的宏参数。




PS:编译过程:
1,扫描解析文件
2,预处理(宏在此时处理,该替换的文字会被替换)
3,对处理过的源代码进行汇编,输出汇编语言的代码(C语言的控制流程被处理)
4,编译为二进制目标文件
5,与程序库进行链接,输出最终的程序文件
(宏 和 C语言在不同的阶段处理执行)

你可能感兴趣的:(c++,C++,宏的高级使用,__VA_ARGS__)