网上学习资料一大堆,但如果学到的知识不成体系,遇到问题时只是浅尝辄止,不再深入研究,那么很难做到真正的技术提升。
一个人可以走的很快,但一群人才能走的更远!不论你是正从事IT行业的老鸟或是对IT行业感兴趣的新人,都欢迎加入我们的的圈子(技术交流、学习资源、职场吐槽、大厂内推、面试辅导),让我们一起学习成长!
前言
大家好啊,我是不一样的烟火a,今天我将会为大家详细讲解其他的预处理指令。虽然本文章读完后不能让大家代码写得飞起,但是预处理这个过程是十分重要的,其中的很多指令也是经常被考到,你了解了它,它将会大幅提升你的C/C++内功,让你学编程更加的容易。
一、#undef
这条指令用于移除一个宏定义。
#undef NAME
//如果现存的一个名字需要被重新定义,那么它的旧名字首先要被移除。
举例:
#include<stdlib.h> // malloc的头文件
#define MALLOC(num, type) (type*)malloc(num * sizeof(type))
int main()
{
// 正常使用
int* p1 = MALLOC(10, int); // 替换后为 int* p1 = (int*)malloc(10 * sizeof(int));
#undef MALLOC // 这条命令下面就用不了MALLOC这个宏了,如果继续用,编译器会报错
char* p2 = MALLOC(5, int); // 这里就会报错:“MALLOC” 未定义
return 0;
}
报错与警告:
二、命令行定义
许多C的编译器提供了一种能力,允许在命令行中定义符号。用于启动编译过程。
提示:当前操作会在Linux环境下演示,vs下演示不出这个效果。
举例:
当我们根据同一个源文件要编译出不同的一个程序的不同版本的时候,这个特性有点用处。(假定某个程序中声明了一个某个长度的数组,如果机器内存有限,我们需要一个很小的数组,但是另外一个机器内存大些,我们需要一个数组能够大些。
#include <stdio.h>
int main()
{
int array[SZ]; // 这里的SZ还没有定义
int i = 0;
for (i = 0; i < SZ; i++)
{
array[i] = i;
}
for (i = 0; i < SZ; i++)
{
printf("%d ", array[i]);
}
printf("\n");
return 0;
}
- 直接用gcc编译上面的代码:
这里gcc就会报错:‘SZ’ undeclared (first use in this function)(’ SZ '未声明(首次在此函数中使用))
- 但是我们可以在编译的时候定义SZ:
//linux 环境演示
gcc -D SZ=10 test.c
运行结果:
也可以在编译的时候将SZ定义为100:
//linux 环境演示
gcc -D SZ=100 test.c
运行结果:
温馨提示:命令行定义的符号也是在预处理阶段将文件中对应符号进行替换的。
三、条件编译
在编译一个程序的时候我们如果要将一条语句(一组语句)编译或者放弃是很方便的。因为我们有条件编译指令。
常见的条件编译指令:
1.
#if 常量表达式(#if后面必须是常量)
//...
#endif
//常量表达式由预处理器求值。
如:
#define __DEBUG__ 1
#if __DEBUG__
//..
#endif
2.多个分支的条件编译
#if 常量表达式
//...
#elif 常量表达式
//...
#else
//...
#endif
3.判断是否被定义
#if defined(symbol) // 全写
#ifdef symbol // 简写
#if !defined(symbol) // 全写
#ifndef symbol // 简写
4.嵌套指令
#if defined(OS_UNIX)
#ifdef OPTION1
unix_version_option1();
#endif
#ifdef OPTION2
unix_version_option2();
#endif
#elif defined(OS_MSDOS)
#ifdef OPTION2
msdos_version_option2();
#endif
#endif
例1(单分支)
#include<stdio.h>
int main()
{
int arr[10] = { 0 };
int i = 0;
for (i = 0; i < 10; ++i)
{
arr[i] = i;
#if 1 // 这里为1代表真,表示下面内容要进入编译
printf("%d ", arr[i]);
#endif // 结束条件编译
}
return 0;
}
编译后的结果:
int main()
{
int arr[10] = { 0 };
int i = 0;
for (i = 0; i < 10; ++i)
{
arr[i] = i;
printf("%d ", arr[i]);
#line 55 "C:\\Users\\86183\\Desktop\\other\\详解预处理\\test.c"
}
return 0;
}
运行结果:
如果改成:
#if 0 // 这里为0代表假,表示下面内容不会进入编译
//...
#endif // 结束条件编译
编译后的结果:
int main()
{
int arr[10] = { 0 };
int i = 0;
for (i = 0; i < 10; ++i)
{
arr[i] = i;
#line 55 "C:\\Users\\86183\\Desktop\\other\\详解预处理\\test.c"
}
return 0;
}
运行结果:
例2(多分支)
#define NUM 2
int main()
{
#if NUM==1
printf("hehe\n");
#elif NUM==2
printf("haha\n");
#elif NUM==3
printf("heihei\n");
#endif
return 0;
}
运行结果:
例3(判断是否被定义)
#define MAX 100
int main()
{
// 全写
#if defined(MAX) // 如果MAX被定义,那么下面语句将会被编译,反之不会。
printf("heihei\n");
#endif
// 简写
#ifdef MAX // 如果MAX被定义,那么下面语句将会被编译,反之不会。
printf("heihei\n");
#endif
// 全写
#if !defined(MAX) // 如果MAX没有被定义,那么下面语句将会被编译,反之不会。
printf("haha\n");
#endif
// 简写
#ifndef MAX // 如果MAX没有被定义,那么下面语句将会被编译,反之不会。
printf("haha\n");
#endif
return 0;
}
运行结果:
例4 (嵌套指令)
#define OS_UNIX 1
#define OPTION2 2
int main()
{
#if defined(OS_UNIX)
#ifdef OPTION1
printf("haha\n");
#endif
![img](https://img-blog.csdnimg.cn/img_convert/0888ff98615603e8dd775f4e9a3ea655.png)
![img](https://img-blog.csdnimg.cn/img_convert/8a5d49dfde7f9993add08560ac09d740.png)
![img](https://img-blog.csdnimg.cn/img_convert/19613ab8ad7a9e25084ee9c565aab722.png)
**既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!**
**由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新**
**[需要这份系统化资料的朋友,可以戳这里获取](https://bbs.csdn.net/topics/618545628)**
)**
>
>
>
#define OS_UNIX 1
#define OPTION2 2
int main()
{
#if defined(OS_UNIX)
#ifdef OPTION1
printf(“haha\n”);
#endif
[外链图片转存中…(img-hlYmECWt-1715635445598)]
[外链图片转存中…(img-d399kQc5-1715635445598)]
[外链图片转存中…(img-Q5pnhfJG-1715635445599)]
既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,涵盖了95%以上大数据知识点,真正体系化!
由于文件比较多,这里只是将部分目录截图出来,全套包含大厂面经、学习笔记、源码讲义、实战项目、大纲路线、讲解视频,并且后续会持续更新