今天在分析一份源码的时候遇到了如下代码:
#define __NR_fork 2
#define _syscall0(type,name) \
type name(void) \
{ \
long __res; \
__asm__ volatile ( "int $0x80" \// 调用系统中断0x80。
:"=a" (__res) \// 返回值??eax(__res)。
:"" (__NR_
##name)); \// 输入为系统中断调用号__NR_name。
if (__res >= 0) \// 如果返回值>=0,则直接返回该值。
return (type) __res; errno = -__res; \// 否则置出错号,并返回-1。
return -1;}
觉得其中的##很有意思,于是就将这个知识点整理一下。#和##都是预处理指令,我们先看#。
#用来把参数转换成字符串,请看下面的两个例子。
例子一:
#include <stdio.h>
#include <iostream>
#define PRINT_STR(s) printf("%s:%d\n",#s,s)
int main(void)
{
int a = 1;
int b = 2;
PRINT_STR(a);
PRINT_STR(b);
PRINT_STR(a+b);
return 0;
}
运行结果:
例子二:
#define SQR(x) printf("The square of x is %d.\n", ((x)*(x)));
如果这样使用宏:SQR(8);
则输出为:The square of x is 64.
注意到没有,引号中的字符x被当作普通文本来处理,而不是被当作一个可以被替换的语言符号。
假如你确实希望在字符串中包含宏参数,那我们就可以使用“#”,它可以把语言符号转化为字符串。上面的例子改一改:
#define SQR(x) printf("The square of "#x" is %d.\n", ((x)*(x)));
再使用:SQR(8);
则输出的是:The square of 8 is 64.
和#运算符一样,##运算符可以用于宏函数的替换部分。这个运算符把两个语言符号组合成单个语言符号。看例子
#define XNAME(n) x ## n
如果这样使用宏:XNAME(8)
则会被展开成这样:x8
##就是个粘合剂,将前后两部分粘合起来,也就是有“字符化”的意思。但是“##”不能随意粘合任意字符,必须是合法的C语言标示符。在单一的宏定义中,最多可以出现一次“#”或“##”预处理操作符。如果没有指定与“#”或“##”预处理操作符相关的计算次序,则会产生问题。为避免该问题,在单一的宏定义中只能使用其中一种操作符(即,一份“#”或一个“##”,或都不用)。除非非常有必要,否则尽量不要使用“#”和“##”。