鹏哥C语言 预处理学习笔记 (#define | # | #undef | 条件编译)
文章参考鹏哥的c语言
一、程序的编译环境和执行环境
可以把test.exe 理解为二进制文件 ,test.c 可以看得懂的
编译环境:vs2022 集成开发环境
执行环境:实际执行代码
bi
二、预编译详解
预定义符号
_FILE_ //进行编译的源文件_LINE_ //文件当前的行号_DATA_ //文件被编译的日期_TIME_ //文件被编译的时间_STDC_//如果编译器遵循ANSIC,其值为1,否则未定义
#define
#define name stuff
(不需要加上分号)
eg. #define MAX 1000
#define reg register
为register这个关键字创建一个简短的名字
#define替换规则
在程序中扩展#define定义符号和宏的步骤:
- 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,他们首先被替换。
- 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被它们的值替换。
- 最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上述处理过程。
注: - 宏参数和#define定义中可以出现其他#define定义的变量,但是对于宏,不能出现递归。
- 当预处理器搜索#define定义的符号的时候,字符串常量的内容并不能被搜索。
#和##
- #:把参数插入到字符串中 ,宏把参数替换到文本中
#include #define PRINT(X) printf("the value of "#X" is %d\n",X)int main(){ int a=10; PRINT(a);//the value of a is 10 int b=20; PRINT(b);//the value of b is 20 int c=30; PRINT(c);//the value of c is 30 return 0;}
- ##:把位于符号两边的符号合成一个符号
#include #define CAT(X,Y) X##Yint main(){ int class101=100; printf("%d\n",CAT(class,101)); // CAT(class,101) == class101 因此输出100 return 0;}
带副作用的宏参数
当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么在使用这个宏时就会出现危险,导致不可预测的后果。副作用就是表达式求值的时候出现的永久性效果(就是看到就会执行)。
x+1;//不带副作用x++;//带有副作用eg.#define MAX(a,b) ((a)>(b)?(a):(b))x=5;y=8;z=MAX(x+1,y+1);//z=((x+1)>(y+1)?(x+1):(y+1)) 只会执行一次y+ 1z=MAX(x++,y++);//z=((x++)>(y++)?(x++):(y++)) 其中y++ 会执行俩次printf("x=%d y=%d z=%d\n",x,y,z);//x=6 y=10 z=9
宏和函数对比
宏通常被应用于执行简单的运算。
命名
我们平时的习惯是,宏名全部大写,函数名不要全部大写
三、#undef
#undef NAME
用于移除一个宏定义
如果现存的一个名字需要被重新定义,那么它的旧名字首先要被移除。
四、条件编译
在编译一个程序时,我们如果要将一条语句编译或放弃时,可以用到条件编译语句。比如,对于调试性的代码,删除可惜保留又碍事,所以我们可以选择性的编译
#include#define __DEBUG__ //如果把这条注释了 就会屏蔽printf中的内容int main(){ int i=0; int arr[10]={0}; for(i=0;i<10;i++) { arr[i]=i; #ifdef __DEBUG__ printf("%d\n",arr[i]); #endif //__DEBUG__ } return 0;}
常用句式
#if 常量表达式 //...#endif如:#define __DEBUG__ 1#if __DEBUG__ //...#endif2、多个分支的条件编译#if 常量表达式 //...#elif 常量表达式 //...#else //...#endif3、判断是否被定义#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
五、其他预处理指令
#error
编译程序时,只要遇到#error
就会生成一个编译错误提示信息,并停止编译#line
改变当前行数和文件名称,它们是在编译程序中预先定义的标识符命令的基本格式:#line number["filename"]
#pragma
它允许向编译程序传送各种指令,支持对程序执行的跟踪。
#pragma 学习链接
#region