个人技术分享

宏定义中的符号粘贴

有些时候,宏参数中的符号并非用来传递数据,而是用来形成多种不同的字串

例如在某些系统函数中,系统本身规范了函数接口的部分标准,形如:

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): 这是宏的参数列表。在这个宏中,有两个参数,分别是numlayer
  • __zinitcall_##layer##_##num: 这是宏的替换部分。在这里,##操作符用于将layernum连接到宏定义的其余部分。

用户的调用如下:

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`。