宏定义中的符号粘贴
有些时候,宏参数中的符号并非用来传递数据,而是用来形成多种不同的字串
例如在某些系统函数中,系统本身规范了函数接口的部分标准,形如:
void __zinitcall_service_1(void)
{
...
}
void __zinitcall_service_2(void)
{
...
}
void __zinitcall_feature_1(void)
{
...
}
void __zinitcall_feature_2(void)
{
...
}
此时,若需要向用户提供一个方便整合字串的宏定义,可以这么写:
#define LAYER_INITCALL(num, layer) __zinitcall_##layer##_##num
#define
: 这是定义预处理器宏的关键字。LAYER_INITCALL
: 这是宏的名称。宏名称通常使用大写字母,以便与变量和函数名区分开来。(num, layer)
: 这是宏的参数列表。在这个宏中,有两个参数,分别是num
和layer
。__zinitcall_##layer##_##num
: 这是宏的替换部分。在这里,##
操作符用于将layer
和num
连接到宏定义的其余部分。
用户的调用如下:
LAYER_INITCALL(service, 1);
LAYER_INITCALL(service, 2);
LAYER_INITCALL(feature, 1);
LAYER_INITCALL(feature, 2);
注意:
在书写非字符串的字串时(如上述例子),使用两边双井号来粘贴字串,并且要注意如果字串出现在最末尾,则最后的双井号必须去除,例如上述代码不可写成:
#define LAYER_INITCALL(num, layer) __zinitcall_##layer##_##num##
但如果粘贴的字串并非出现在最末尾,则前后都必须加上双井号:
#define LAYER_INITCALL(num, layer) __zinitcall_##layer##_##num##end
注意:
另外,如果字串本身拼接为字符串,那么只需要使用一个井号即可,比如:
#define domainName(a, b) "www." #a "." #b ".com"
int main()
{
printf("%s\n", domainName(yueqian, lab));
}
执行打印如下:
gec@ubuntu:~$ ./a.out
www.yueqian.lab.com
gec@ubuntu:~$
当使用`#`操作符时,标识符会被转换为字符串字面量,而`##`操作符用于连接两个标识符。
例如:
#define STR(s) #s
#define CONCAT(x, y) x##y
printf("%s\n", STR(Hello)); // 输出 "Hello"
int CONCAT(num, 1) = 10; // 这将创建一个名为 num1 的整型变量
在这里,`STR(Hello)`会被替换为`"Hello"`,而`CONCAT(num, 1)`会被替换为`num1`。