关于GCC和C99中可变参数宏
2014-03-12 21:57
218 查看
参考文章:http://blog.csdn.net/woshinia/article/details/8255693
首先介绍一下预处理连接符“##”,其作用是将两个符号连接成一个,即当可变参数__VA_ARGS__的个数为0时,##负责把前面多余的‘,’去掉,从而防止编译错误(error: expected expression before ')' token)
还有一些常用的调试打印时用到的宏如下:
1) __VA_ARGS__ 是一个可变参数的宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)
2) __FILE__ 宏在预编译时会替换成当前的源文件名
3) __LINE__宏在预编译时会替换成当前的行号
4) __FUNCTION__宏在预编译时会替换成当前的函数名称
参考文章中的博主搜集了很多关于宏的资料,有兴趣可以看一看,部分摘抄如下:
[b]1.#
假如希望在字符串中包含宏参数,ANSI C允许这样作,在类函数宏的替换部分,#符号用作一个预处理运算符,它可以把语言符号转化程字符串。例如,如果x是一个宏参量,那么#x可以把参数名转化成相应的字符串。该过程称为字符串化(stringizing).
#incldue <stdio.h>
#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 <stdio.h>
#define XNAME(n) x##n
#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;
}
输出结果:
x1=12
3.可变参数宏 ...和_ _VA_ARGS_ _
__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串。比如:
#define PR(...) printf(__VA_ARGS__)
int main()
{
int wt=1,sp=2;
PR("hello\n");
PR("weight = %d, shipping = %d",wt,sp);
return 0;
}
输出结果:
hello
weight = 1, shipping = 2
省略号只能代替最后面的宏参数。
#define W(x,...,y)错误![/b]
首先介绍一下预处理连接符“##”,其作用是将两个符号连接成一个,即当可变参数__VA_ARGS__的个数为0时,##负责把前面多余的‘,’去掉,从而防止编译错误(error: expected expression before ')' token)
还有一些常用的调试打印时用到的宏如下:
1) __VA_ARGS__ 是一个可变参数的宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)
2) __FILE__ 宏在预编译时会替换成当前的源文件名
3) __LINE__宏在预编译时会替换成当前的行号
4) __FUNCTION__宏在预编译时会替换成当前的函数名称
参考文章中的博主搜集了很多关于宏的资料,有兴趣可以看一看,部分摘抄如下:
[b]1.#
假如希望在字符串中包含宏参数,ANSI C允许这样作,在类函数宏的替换部分,#符号用作一个预处理运算符,它可以把语言符号转化程字符串。例如,如果x是一个宏参量,那么#x可以把参数名转化成相应的字符串。该过程称为字符串化(stringizing).
#incldue <stdio.h>
#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 <stdio.h>
#define XNAME(n) x##n
#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;
}
输出结果:
x1=12
3.可变参数宏 ...和_ _VA_ARGS_ _
__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏_ _VA_ARGS_ _就可以被用在替换部分中,替换省略号所代表的字符串。比如:
#define PR(...) printf(__VA_ARGS__)
int main()
{
int wt=1,sp=2;
PR("hello\n");
PR("weight = %d, shipping = %d",wt,sp);
return 0;
}
输出结果:
hello
weight = 1, shipping = 2
省略号只能代替最后面的宏参数。
#define W(x,...,y)错误![/b]
相关文章推荐
- 关于GCC和C99中可变参数宏
- 用GCC和C99的可变参数宏, 更方便地打印调试信息
- 关于printk可变参数
- 关于C/C++中可变参数的详细介绍(va_list,va_start,va_arg,va_end)
- 关于Python参数传递时,传递可变对象(mutable)和不可变更对象(immutable)的误区
- 关于iphone可变参数函数的定义
- Java中关于可变长参数的那些事[参数中使用省略号的情况]
- 关于gcc编译参数 -idirafter
- 关于C#的可变长参数
- 关于可变参数
- c99 可变参数宏 __VA_ARGS__
- GCC下,C语言参数的内存分配 以及 函数可变参数
- C#关于静态函数与普通函数之间的调用问题,以及可变参数的使用
- 关于可变参数列表的总结
- 转:关于GCC中同时使用动态和静态库链接的操作参数和解释
- 彤姐的程序篇(四)关于支持可变参数的函数.
- C语言关于可变参数函数的例子
- 关于可变参数(不定参数),以及windows下将进程的cpu,内存统计信息写到日志
- 关于python函数的默认参数,可变参数,关键字参数
- 关于Java可变参数的若干问题