【C语言进阶剖析】21、宏定义与使用分析


C 语言中的宏定义

  • #define 是预处理器处理的单元实体之一
  • #define 定义的宏可以出现在程序的任意位置
  • #define 定义之后的代码都可以使用这个宏

1 定义宏常量

  • #define 定义的宏常量可以直接使用
  • #define 定义的宏常量本质为字面量

下面定义的宏正确吗?
在这里插入图片描述
代码如下:

#define ERROR -1
#define PATH1 "D:\test\test.c" 
#define PATH2 D:\test\test.c
#define PATH3 D:\test\
test.c
int main()
{
    int err = ERROR;
    char* p1 = PATH1;
    char* p2 = PATH2;
    char* p3 = PATH3;
}

先进行预编译

$ gcc -E 21-1.c -o 21-1.i

预编译通过,打开 21-1.i 文件,如下所示:

# 1 "21-1.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 31 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 32 "<command-line>" 2
# 1 "21-1.c"





int main()
{
    int err = -1;
    char* p1 = "D:\test\test.c";
    char* p2 = D:\test\test.c;
    char* p3 = D:\testtest.c;
}

宏定义被直接文本替换,我们定义的 #define 消失了,可以看出,p2,p3 显然指向的字符串不符合 C 语言的语法结构,是错误的,预处理只是进行文本替换,不负责类型检查,真正查找错误还靠编译阶段。

小贴士:宏定义不占内存空间,因为宏在预处理阶段就会被替换掉,到了编译的阶段是没有宏存在的,自然到不了可执行文件中,所以它不占内存空间。

2 宏定义表达式

  • #define 表达式的使用类似函数调用
  • #define 表达式可以比函数更强大,但是比函数更容易出错

为什么说宏很强大,宏可以做很多函数做不到的事,比如求解一个数组的长度;得出当前文件名,当前行号;将参数类型作为参数传入等等,这些函数都办不到,后面我们一一说明

下面的宏表达式定义正确吗?
在这里插入图片描述

// 21-2.c
#include<stdio.h>
#define _SUM_(a, b) (a) + (b)
#define _MIN_(a, b) ((a) < (b) ? (a) : (b))
#define _DIM_(a) sizeof(a)/sizeof(*a)
int main()
{
    int a = 1;
    int b = 2;
    int c[4] = {0};
    int s1 = _SUM_(a, b);
    int s2 = _SUM_(a, b) * _SUM_(a, b);
    int m = _MIN_(a++, b);
    int d = _DIM_(c);
    printf("s1 = %d\n", s1);
    printf("s2 = %d\n", s2);
    printf("m = %d\n", m);
    printf("d = %d\n", d);
    return 0;
}

先单步编译,只进行预处理,编译通过;在进行编译,编译也通过,运行,结果如下:

$ gcc -E 21-2.c -o 21-2.i
$ gcc 21-2.c -o 21-2
$ ./21-2
s1 = 3
s2 = 5
m = 2
d = 4

上面的代码可以看出,s2 是求 a 和 b 和的平方,结果是 5;m 是 a++ 和 b 的最小值,结果为 2,这和我们预想的结果不太对,为什么会出现这种情况呢,我们来看看预处理之后的文件(这里为了省略不必要的信息,注释了 #include<stdio.h> 再执行的预处理)

# 1 "21-2.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 31 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 32 "<command-line>" 2
# 1 "21-2.c"





int main()
{
    int a = 1;
    int b = 2;
    int c[4] = {0};
    int s1 = (a) + (b);
    int s2 = (a) + (b) * (a) + (b);
    int m = ((a++) < (b) ? (a++) : (b));
    int d = sizeof(c)/sizeof(*c);
    printf("s1 = %d\n", s1);
    printf("s2 = %d\n", s2);
    printf("m = %d\n", m);
    printf("d = %d\n", d);
    return 0;
}

可以看到,

  • s2 = (a) + (b) * (a) + (b) = 1 + 2 * 1 + 2 = 5
  • m = ((a++) < (b) ? (a++) : (b)),执行 (a++) < (b) 结果为真,选择(a++) ,此时 a 的值为 2,所以结果为 2
  • 使用宏可以求出数组长度,这在 c 语言中用函数是一定办不到的。

上面我们看到了宏表达式的副作用,在使用时一定要小心。

3 宏表达式与函数表达式的对比

  • 宏表达式被预处理器处理,编译器不知道宏表达式的存在
  • 宏表达式用 “实参” 完全替代形参,不进行任何运算(完全是文本替换)
  • 宏表达式没有任何的 “调用” 开销
  • 宏表达式中不能出现递归定义

比如下面的表达式就是不正确的,宏不能递归定义
在这里插入图片描述

3 宏有作用域吗?

有个有趣的问题:宏定义的常量或表达式是否有作用于限制?

看下面的程序合法吗?在函数 def() 中定义了宏,能在函数 area() 中使用吗?
在这里插入图片描述

// 21-3.c
#include<stdio.h>
void def()
{
    #define PI 3.1415926
    #define AREA(r) r * r *PI
}
double area(double r)
{
    return AREA(r);
}
int main()
{
    double r = area(5);
    printf("PI = %f\n", PI);
    printf("d = 5; a = %f\n", r);
    return 0;
}

上面的程序编译运行通过,结果如下:
在这里插入图片描述
为什么在函数 def() 中定义的宏能在函数 area() 中使用吗?

#define 定义之后的代码都可以使用这个宏,C 语言中作用域的概念的是针对变量和函数的,不针对宏,因为宏被预处理器处理,编译器根本就不知道宏的存在,所以没法将作用域的概念加到宏上面。

我们再来看一下单步编译的结果

$ gcc -E 21-3.c -o 21-3.i
# 1 "21-3.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 31 "<command-line>"
# 1 "/usr/include/stdc-predef.h" 1 3 4
# 32 "<command-line>" 2
# 1 "21-3.c"


void def()
{


}
double area(double r)
{
    return r * r *3.1415926;
}
int main()
{
    double r = area(5);
    printf("PI = %f\n", 3.1415926);
    printf("d = 5; a = %f\n", r);
    return 0;
}

4 强大的内置宏

下面我们来看看 C 语言中强大的内置宏,可以查看被编译的文件名,当前行号,被编译的日期等等。

含义实例
__FILE__被编译的文件名file.c
__LINE__当前行号25
__DATE__编译时的日期Nov 18 2019
__TIME__编译时的时间21:01:01
__STDC__编译器是否遵循标准 C 规范1

下面再来看一个示例:宏使用的综合示例

// 21-4.c
#include<stdio.h>
#include<malloc.h>

#define MALLOC(type, x) (type*)malloc(sizeof(type)*x)
#define FREE(p) (free(p), p = NULL)				// 逗号表达式
#define LOG(s) printf("[%s]{%s:%d} %s \n", __DATE__, __FILE__, __LINE__, s)

#define FOREACH(i, m) for(i = 0; i < m; i++)
#define BEGIN {
#define END   }

int main()
{
    int x = 0;
    int* p = MALLOC(int, 5);
    LOG("Begin to run main code...");
    FOREACH(x, 5)
    BEGIN
        p[x] = x;
    END
    FOREACH(x, 5)
    BEGIN
        printf("%d\n", p[x]);
    END
    FREE(p);
    LOG("End");
    return 0;
}

在这里插入图片描述

  • MALLOC 的作用是在堆空间中申请一个数组,需要注意的是,这里可以将参数类型作为自变量传递给宏
  • FREE 是用来释放数组的。通过这个宏还可以在释放空间的同时将指针置为空
  • FOREACH 宏也是宏比函数强大的地方,在函数中没有办法实现这个概念
  • 主函数先申请一个数组,再将数组中的元素赋值,最后将数组元素打印出来

5 小结

  • 预处理器直接对宏进行文本替换,参数不会进行求值、运算
  • 预处理器不会对宏定义进行语法检查,语法错误只能被编译器检测
  • 宏的效率高于函数调用,也会带来一定的副作用
  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值