1、c语言编译过程
gcc -E hello.c -o hello.i 1、预处理
gcc -S hello.i -o hello.s 2、编译
gcc -c hello.s -o hello.o 3、汇编
gcc hello.o -o hello_elf 4、链接
预编译:将.c中的头文件展开、宏展开,生成的文件是.i文件。
编译:将预处理之后的.i文件生成.s汇编文件。
汇编:将.s汇编文件生成.o目标文件。
链接:将.o文件链接成目标文件。
2、include
#include<>//用尖括号包含头文件,在系统指定的路径下找头文件
#include""//用双引号包含头文件,先在当前目录下找头文件,找不到再到系统指定的路径下找。
注意:include经常用来包含头文件,可以包含.c文件。但是由于include包含的文件会在预编译被展开,如果一个.c被包含多次,展开多次,会导致函数的重复定义。
注意:预处理只对include等预处理操作进行处理并不会进行语法检查,这个阶段有语法错误也不会报错,第二个阶段即编译阶段才会进行语法检查。
3、define
定义宏用define去定义。宏是在预编译的时候进行替换。
3.1、不带参宏
#define PI 3.14
在预编译的时候如果代码中出现了PI就用3.14去替换。
宏的好处:只要修改宏定义,其他地方在预编译的时候就会重新替换。
注意:宏定义后边不要加分号。
宏定义的作用范围::从定义的地方到本文件末尾。
如果想在中间终止宏的定义范围:#undef PI//终止PI的作用
3.2、带参宏
#define S(a,b) a*b
注意带参宏的形参a和b没有类型名。宏定义只是简单的替换,不会进行加括号等操作。
比如:S(2+8,4)=34;//2 + 8 * 4 = 34;
3.3、带参宏和带参函数的区别
带参宏被调用多少次就会展开多少次,执行代码的时候没有函数调用的过程,不需要压栈弹栈。所以带参宏是浪费了空间,因为被展开多次,节省时间。
带参函数,代码只有一份,存在代码段,调用的时候去代码段取指令,调用的时候要压栈弹栈。有个调用的过程。所以说带参函数浪费了时间节省了空间。
带参函数的形参有类型名,带参宏的形参没有类型名。
3.4、选择性编译
(1)、
#ifdef AAA
代码段一
#else
代码段二
#endif
如果在当前.c ifdef上边定义过AAA,就编译代码段一,否则编译代码段二。
注意和if else语句的区别,if else语句都会被编译,通过条件选择性的执行代码而选择性编译只有一块代码被编译。
(2)、
#ifndef AAA
代码段一
#else
代码段二
#endif
和第一种互补,经常用于防止头文件重复包含。
(3)、
#if 表达式一
代码段一
#else
代码段二
#endif
如果表达式为真,编译第一段代码,否则编译第二段代码。
选择性编译都是在预编译阶段干的事情。