13 预处理器
13.1 基本介绍
13.1.1 预处理器
宏替换、文件包含、条件编译
13.1.2 预处理指令
预处理指令都以 # 开头,结尾不能使用分号,不能写在函数的内部,必须写在开头。
13.2 宏定义
13.2.1 基本介绍
语法:
#define 宏名称 替换文本
13.2.2 使用宏定义常量
宏定义定义的常量实在预处理阶段将宏定义纯文本替换,const函数实在编译阶段,要进行类型检测才会进行纯文本替换,所以说在初始化数组时,数组的大小只能使用define宏定义来进行替换,并不能使用const函数,因为数组和const一起进行编译,编译器并不会认为他是一个常量,建议在数组的初始化时并不定义数组的大小。
13.2.3 使用宏定义数据类型
13.2.4 宏定义的替换文本
13.2.5 宏定义嵌套
13.2.6 取消宏定义
#undef 宏名称
13.3 带参数的宏定义
13.3.1 基本介绍
宏名(实参列表)
#include <stdio.h>
#define MAX(a, b) (a > b) ? a : b
int main()
{
int x, y, max;
printf("input two numbers: ");
scanf("%d %d", &x, &y);
// 说明
// 1. MAX(x, y); 调用带参数宏定义
// 2. 在宏替换时(预处理,由预处理器), 会进行字符串的替换,同时会使用实参, 去替换形参
// 3. 即MAX(x, y) 宏替换后 (x>y) ? x : y
max = MAX(x, y);
printf("max=%d\n", max);
return 0;
}
13.3.2 注意事项和细节
带参宏定义中,形参之间可以出现空格,但是宏名和形参列表之间不能有空格出现。
在带参宏定义中,不会为形式参数分配内存,因此不必指明数据类型,而在宏调用中,实参包含了具体的数据,要用它们去替换形参,因此实参必须要指明数据类型。
在宏定义中,替换文本内的形参通常要用括号括起来以避免出错。
13.3.3 带宏定义和函数的区别
宏定义只是简单的替换,在比那一之前就处理了,函数时会编译的。
13.4 文件包含
标准库头文件、自定义头文件的扩展名都是 .h。
标准头文件引入时用< >
自定义头文件引入时用" "
13.4.1 包含标准库头文件
13.4.2 包含自定义头文件
可以引入相对路径或者绝对路径
#include "C:\Preparation\Embedded\01CLang\code\project\foo.h"
#include "/usr/local/lib/foo.h"
13.5 条件编译
13.5.1 条件编译
#if 表达式 //弱国表达式不为0,为真,执行语句,为0则忽略语句
执行语句
#endif
#if 表达式
执行语句
#else
执行语句
#endif
#if 表达式
执行语句
#elif 表达式
执行语句
#else
执行语句
#endif
13.5.2 #ifdef
#ifdef 宏定义名称
printf
#denif
用来检查宏定义是否定义过,如果存在就表示加载过库文件,就会打印一行提示。
13.5.3 #if defined
与#ifdef一样
#if defined FOO
x = 2;
#elif defined BAR
x = 3;
#else
x = 4;
#endif
13.5.4 #ifndef
z正好与前面两个相反
用来判断哪一个宏没有被定义过
13.3.5 应用案例
开发一个C语言程序,让它暂停 5 秒以后再输出内容"helllo, 尚硅谷!~",并且要求跨平台,在Windows和Linux下都能运行,如何处理。
#include <stdio.h>
#if _WIN32 //如果是windows平台, 就引入 <windows.h>
#include <windows.h>
#define SLEEP(t) Sleep(t * 1000)
#elif __linux__ // 如果是linux平台, 就引入<unistd.h>
#include <unistd.h>
#define SLEEP sleep
#endif
int main()
{
SLEEP(5);
printf("hello, 尚硅谷~");
return 0;
}
13.6 命令总结
指令 | 说明 |
---|---|
#include | 包含一个源文件代码 |
#define | 宏定义 |
#undef | 取消宏定义 |
#if | 若为真执行代码 |
#ifdef | 如果定义这个宏,执行代码 |
#ifndef | 如果没有定义这个宏,执行代码 |
#elif | 如果#if为假,当前条件为真,执行语句 |
#endif | 结束 |