#define 和 do{...}while(0) 的妙用

没有检索到摘要

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >


最近在阅读公司代码的时候发现宏定义 #define 后面的语句都是被 do{…}while(0) 圈起来的。作为一个还没毕业的实习生,对此很是不解。

正确认识宏定义#define

#define是在预处理的时候进行直接替换
这个是重点!
举例:

#define func(x) x*10-2
int res = 4*func(3);

res 的结果是多少? 算一下。结果应该是 118 。千万别算成 112 。
仔细理解一下上面黄色字体的内容,很好理解的。
因为上面编译后是这样的:

int res = 4*3*10-2;

如果定义的是一个函数会怎么样;

void print1() {
    printf("Hello 1");
}
void print2() {
    printf("Hello 2");
}
#define func() print1();print2();
int main() {
    if (0) 
        func();
    return 0;
}

会不会有输出?
答案是有的。 再到前面看一下黄色字体,#define是在预处理的时候进行直接替换。
是不是理解了,会输出 hello 2
上面例子编译后main 函数是这样的:

int main() {
    if (0) 
        print1();
        print2();
    return 0;
}

我虽然把print1(), print2()写在了共同的缩进格式,但是C语言不是python,有点 C 基础的人都应该懂得,if 语句的范围只限于 print1();print2();是在 if 条件之外的,因此会被输出。
对的,#define 在编译的时候会被直接替换。
当然如果main 函数里面这么写就没错了。

int main() {
    if (0) {
        func();
    }
    return 0;
}

这样符合我们的逻辑。但是这样不是自己给自己后期增加烦恼吗,万一忘记了呢,后期维护可能是很多人在做,不能保证每个人都这么小心。

do{…}while(0) 的妙用

但是在公司一个大项目中,对于几十上百万行的代码,我们很难去管理上面的错误。可能一个项目会有多个人参与,好多人维护,如果把经历花费在维护这样的事情上当然很不值得,因此 do{...}while(0);的,妙用就体现出来了。
我们可以自己待到上面的例子看看:

void print1() {
    printf("Hello 1");
}
void print2() {
    printf("Hello 2");
}
#define func()               \
     do {                    \
         print1();print2();  \
     } while(0)
int main() {
    if (0) 
        func();
    return 0;
}

编译后main函数是这样的:

int main() {
    if (0) 
         do {                
             print1();print2(); 
         } while(0);
    return 0;

这样就不会出现编译错误。

评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值