C语言中的#和##操作符用法

C语言中#和##操作符用法

答:在C语言中,###是预处理器(preprocessor)的操作符,主要用于宏(macro)的定义中。这两个操作符提供了字符串化和字符串连接的功能。

#操作符

#操作符用于将其后的宏参数转换为一个字符串字面量,在编译时将宏参数转换为字符串。

下面根据一个简单的示例分析一下。

c 复制代码
#include <stdio.h>

#define STRINGIFY(str) #str

int main(void)
{
    printf("STRINGIFY(hello)=%s\n", STRINGIFY(hello));

    return 0;
}

执行如下预处理命令:

bash 复制代码
gcc -E main.c -o main.i

查看main.i文件,可以发现确实将宏转换为了字符串:

c 复制代码
...
# 5 "main.c"
int main(void)
{
    printf("STRINGIFY(hello)=%s\n", "hello");

    return 0;
}

编译成可执行文件,运行测试:

bash 复制代码
$ gcc main.c -o main
$ ./main 
STRINGIFY(hello)=hello

##操作符

##操作符将两个标识符连接在一起,在编译时进行标识符连接。

下面根据一个简单的示例分析一下。

c 复制代码
#include <stdio.h>

#define CONCAT(x, y) x##y

int main(void)
{
    int ab = 1;
    
    printf("CONCAT(a,b)=%d\n", CONCAT(a,b));

    return 0;
}

执行如下预处理命令:

bash 复制代码
gcc -E main.c -o main.i

查看main.i文件,可以发现确实将宏参数进行了连接:

c 复制代码
...
# 5 "main.c"
int main(void)
{
    int ab = 1;

    printf("CONCAT(a,b)=%d\n", ab);

    return 0;
}

编译成可执行文件,运行测试:

bash 复制代码
$ gcc main.c -o main
$ ./main 
CONCAT(a,b)=1

注意

如果宏参数也是另外一个宏时,比如上面的str或者xy也是定义的一个宏:

c 复制代码
#include <stdio.h>

#define STRINGIFY(str) #str
#define CONCAT(x, y) x##y

#define HELLO hello
#define A   a
#define B   b

int main(void)
{
    int ab = 1;
    
    printf("STRINGIFY(hello)=%s\n", STRINGIFY(HELLO));
    printf("CONCAT(a,b)=%d\n", CONCAT(A,B));

    return 0;
}

上面STRINGIFY(HELLO)就直接展开为大写字符串HELLO,而 CONCAT(A,B)会被展开为AB,导致编译失败,找不到变量AB,这就说明当宏参数是另一个宏的时候,宏定义里有用###的地方宏参数是不会再展开。

c 复制代码
...
# 10 "main.c"
int main(void)
{
    int ab = 1;

    printf("STRINGIFY(hello)=%s\n", "HELLO");
    printf("CONCAT(a,b)=%d\n", AB);

    return 0;
}

解决办法是再定义一层中间转换宏,保证所有的宏参数在这个转换宏中展开,如下:

c 复制代码
#include <stdio.h>

#define _STRINGIFY(str) #str
#define STRINGIFY(str) _STRINGIFY(str)

#define _CONCAT(x, y) x##y
#define CONCAT(x, y) _CONCAT(x, y)

#define HELLO hello
#define A   a
#define B   b

int main(void)
{
    int ab = 1;
    
    printf("STRINGIFY(hello)=%s\n", STRINGIFY(HELLO));
    printf("CONCAT(a,b)=%d\n", CONCAT(A,B));

    return 0;
}

编译预处理即可按照期望进行展开:

c 复制代码
...
# 13 "main.c"
int main(void)
{
    int ab = 1;

    printf("STRINGIFY(hello)=%s\n", "hello");
    printf("CONCAT(a,b)=%d\n", ab);

    return 0;
}

编译成可执行文件,运行测试:

bash 复制代码
$ gcc main.c -o main
$ ./main 
STRINGIFY(hello)=hello
CONCAT(a,b)=1
相关推荐
疯狂的挖掘机6 小时前
记一次基于QT的图片操作处理优化思路(包括在图上放大缩小,截图,画线,取值等)
开发语言·数据库·qt
cnxy1886 小时前
围棋对弈Python程序开发完整指南:步骤4 - 提子逻辑和劫争规则实现
开发语言·python·机器学习
意趣新6 小时前
C 语言源文件从编写完成到最终生成可执行文件的完整、详细过程
c语言·开发语言
李艺为7 小时前
根据apk包名动态修改Android品牌与型号
android·开发语言
黄河滴滴8 小时前
java系统变卡变慢的原因是什么?从oom的角度分析
java·开发语言
老华带你飞8 小时前
农产品销售管理|基于java + vue农产品销售管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端
superman超哥8 小时前
Rust Workspace 多项目管理:单体仓库的优雅组织
开发语言·rust·多项目管理·rust workspace·单体仓库
kylezhao20198 小时前
C#通过HSLCommunication库操作PLC用法
开发语言·c#
lengjingzju9 小时前
一网打尽Linux IPC(三):System V IPC
linux·服务器·c语言
JIngJaneIL9 小时前
基于springboot + vue房屋租赁管理系统(源码+数据库+文档)
java·开发语言·前端·数据库·vue.js·spring boot·后端