互联网科技嵌入式编程

奇怪的宏

2020-05-01  本文已影响0人  罗蓁蓁

这一篇介绍这些奇怪的宏。

一、do while(0)

为了交换两个整型变量的值,前面值传递中已经用包含指针参数的 swap 函数做到了,这次用来实现(swap.c):

#include <stdio.h>
#define SWAP(a,b)       \
    do{                 \
        int t = a;      \
        a = b;          \
        b = t;          \
    }while(0)

int main()
{
    int c=1, d=2;
    int t;  // 测试 SWAP 与环境的兼容性
    
    SWAP(c,d);
    
    printf("c:%d d:%d\n", c, d);
    return 0;
}

这个宏看起来就有点怪了:do while(0) 是写了个循环又不让它循环,蛋疼啊!其实不然,这样写是有妙用的:

首先,SWAP 有多条语句,如果这样写:

#define SWAP(a,b)       \
        int t = a;      \
        a = b;          \
        b = t;

那么用的时候就得这么用:

SWAP(c,d)

不能加分号!加了很难看!不习惯吧?

其次,使用 do{...}while(0),中间的语句用大括号括起来了,所以是另一个命名空间,其中的新变量 t 不会发生命名冲突

SWAP 宏要比之前那个函数的效率要高,因为没有发生函数调用,没有参数传递,宏会在编译前被替换,所以只是嵌入了一小段代码。

二、##

标题我没打错,这里要说的就是井号,#的功能是将其后面的宏参数进行字符串化操作。比如下面代码中的宏:

#define WARN_IF(EXP) \
do{ if (EXP) \
    fprintf(stderr, "Warning: " #EXP "\n"); } \
while(0) 

那么实际使用中会出现下面所示的替换过程:

WARN_IF (divider == 0); 

被替换为 :

do { if (divider == 0) 
    fprintf(stderr, "Warning: " "divider == 0" "\n"); 
} while(0); 

需要注意的是:C语言中多个双引号字符串放在一起会自动连接起来,所以如果 divider 为 0 的话,就会打印出:

Warning: divider == 0

三、####

# 还是比较少用的,## 却比较流行,在 linux0.01 中就用到过。##被称为连接符,
用来将两个 记号(编译原理中的词汇) 连接为一个 记号。

看下面的例子吧(add.c):

#include <stdio.h>

#define add(Type)               \
Type add##Type(Type a, Type b){ \
    return a+b;                 \
}

// 下面两条是奇迹发生的地方
add(int)
add(double)

int main()
{
    int a = addint(1, 2);
    double d = adddouble(1.5, 1.5);
    
    printf("a:%d d:%lf\n", a, d);
    return 0;
}

那两行被替换后是这个样子的:

int addint(int a, int b){ return a+b; }
double adddouble(double a, double b){ return a+b; }

出差必备

买火车票、高铁票、机票,订酒店都打9折的出行工具TRIP,点击注册

上一篇 下一篇

猜你喜欢

热点阅读