![d1ef730f599c462a31dfa21349b385c2.png](https://i-blog.csdnimg.cn/blog_migrate/e89a19343870e4ec886605556de4dfa5.png)
![ae19188111e26b6528cec61b0528ed2e.png](https://i-blog.csdnimg.cn/blog_migrate/44fdb94e3491ddc48a4532e4d76ff082.png)
如今物联网行业兴起,许多小伙伴都对学习能够软硬件兼备的C语言比较感兴趣,相信大家学习C语言的时候都是从“hello world”开始的:
#include <stdio.h>
int main()
{
printf(“hello worldn”);
return 0;
}
第一步包含头文件,第二步定义出主函数,第三 步在函数内部打印hello world
今天我们要聊的内容就是这第一步,到底为什么要包含头文件,而这条语句又是在那个时间点实现的呢?
在主函数中我们需要使用printf函数,而这个函数的定义与声明都是在stdio.h这个标准输入输出库中完成的,这个库中包含的就是与输入输出相关的函数,例如scanf、gchar、put等等,我们的代码要包含这个库,才能够找到printf函数,能够正常使用。
连接到这个库就是在预处理环节实现的,很多小伙伴知道我们的代码运行出效果之前需要编译,但其实编译是有4个环节的:
gcc -E hello.c -o hello.i 1、预处理
gcc -S hello.i -o hello.s 2、编译
gcc -o hello.s -o hello.o 3、汇编
gcc hello.o -o hello 4、链接
1:预处理
将.c中的头文件展开 、宏展开->生成的文件i文件
2:编译
将预处理后的.i文件 生成.s汇编文件
3:汇编
将.s汇编文件生成.o目标文件
4:链接
将.o文件链接成目标文件
汇总为一就是:
gcc hello.c -o hello
大家可以看到包含头文件就是在预处理环节完成的,所谓预处理是指在进行编译的第一遍扫描(词法扫描和语法分析)之前所作的工作。预处理是C语言的一个重要功能, 它由预处理程序负责完成。当对一个源文件进行编译时,系统将自动引用预处理程序对源程序中的预处理部分作处理,处理完毕自动进入对源程序的编译。那么包括包含头文件在内的还有那些工作是在预处理环节完成的呢?
(1)include
#include <> 用尖括号包含的头文件,在系统指定的路径下找头文件
#include "" 用双引号包含的头文件,现在当前目录下找头文件,找不到,再到系统指定路径下找
注意:include 常用来包含头文件,可以包含.c文件,但不要多次包含
include 包含的文件会在预处理的时候被展开,如果一个.c被包含多次,展开多次,会导致重复定义
注意:预处理的时候只是对include 的预处理操作进行处理,并不会进行语法的检查
这个阶段,有语法错误也不会报错,第二个阶段(编译阶段)再进行语法检查
(2)define
定义宏用define 去定义
宏是在预编译的时候进行替换
①不带参宏
#define PI 3.14
注意:宏定义后边不要加分号
![521c03748a7fedf09cea9d0f73ac64b6.png](https://i-blog.csdnimg.cn/blog_migrate/c033a43e9f707d7e3aba62f48f422be1.png)
在预编译的时候,如果代码里出现了PI 就用3.14去替换
宏的好处:只要修改宏定义,其他地方在预编译的时候就会重新替换。
![0a3e9ff8dffffbd4fb3ed8734b9d061f.png](https://i-blog.csdnimg.cn/blog_migrate/7caa5c1c8a1e71e6cf41eb5ac4bd641b.png)
-0.000000
预处理只替换PI
赋值在运行之后
宏定义的范围:从定义的地方,直到本文件的末尾。
如果想在中间终止宏定义的范围
#undef PI //终止PI的作用
![1738939f874408a5115361067a7f760a.png](https://i-blog.csdnimg.cn/blog_migrate/441d4184b851eaee7b43aad4c0512791.png)
②带参宏
# define S(a,b) a*b
注意带参宏的形参没有类型名,
S(2,3)将来在预处理的时候 实参代替字符串的形参,其他数字保留2*3
S(2+4,3)被替换成2+4*3
![d039a825e39efb7a1f9eb25d1bcc8ebe.png](https://i-blog.csdnimg.cn/blog_migrate/e2eaf9026c2acdaaf524a935c4095122.png)
注意:带参宏,是预处理的时候进行替换
解决歧义方法
![0d26eefdbd9168e66367f3154345afc2.png](https://i-blog.csdnimg.cn/blog_migrate/91e12bf5fdd5cc3f10e506b1fcc1011e.jpeg)
③带参宏和带参函数的区别
带参宏被调用多少次就会展开多少次,执行代码的时候没有函数被调用过程,浪费了空间,节省时间
带参函数,代码只有一份,存放在代码区,调用的时候,有一个过程,浪费时间,节省了空间
带参函数的形参是有类型的,带参宏的形参没有类型。
(3)选择性编译
①
#ifdef AAA
代码段一
#else
代码段二
#endif
如果在当前.c ifdef 上边定义过AAA,就编译代码段一,否则,编译代码段二
注意: if else 语句都会被编译,
通过条件选择性执行的代码 已有一块代码被编译
![a80ff2a16fae1c7d85c26bda9bd85b27.png](https://i-blog.csdnimg.cn/blog_migrate/909db5c557253221914c376f03907aa0.jpeg)
②
#ifndef AAA
代码段一
#else
代码段二
#endif
与第一种互补
这种方法,经常用来防止头文件重复包含
③
#if 表达式
程序段一
#else
程序段二
#endif
如果表达式为真,编译第一段代码,否则编译第二段代码
选择性编译都是在预编译阶段做的事情
![9da01445d1e47cc429540a5f00a3f952.png](https://i-blog.csdnimg.cn/blog_migrate/c5f1d1912b1911775d9f62132232df38.jpeg)
重点小结:
- 预处理功能是C语言特有的功能,它是在对源程序正式编译前由预处理程序完成的。程序员在程序中用预处理命令来调用这些功能。
2. 宏定义是用一个标识符来表示一个字符串,这个字符串可以是常量、变量或表达式。在宏调用中将用该字符串代换宏名。
3. 宏定义可以带有参数,宏调用时是以实参代换形参。而不是“值传送”。
4. 为了避免宏代换时发生错误,宏定义中的字符串应加括号,字符串中出现的形式参数两边也应加括号。
5. 文件包含是预处理的一个重要功能,它可用来把多个源文件连接成一个源文件进行编译,结果将生成一个目标文件。
6. 条件编译允许只编译源程序中满足条件的程序段,使生成的目标程序较短,从而减少了内存的开销并提高了程序的效率。
7. 使用预处理功能便于程序的修改、阅读、移植和调试,也便于实现模块化程序设计。