以hello.c文件为例:
从 高级语言->汇编语言->机器语言(二进制)
-
预处理编译器:cpp gcc -E hello.c -o hello.i 头文件展开,宏替换,注释去掉
-
编译器:gcc gcc -S hello.i -o hello.s c文件变成汇编文件
-
汇编器:as gcc -c hello.s -o hello.o 汇编文件变成二进制文件
-
链接器:ld gcc hello.o -o hello 将函数库中相应的代码组合到目标文件中
第一步预处理
由.c文件到.i文件,这个过程叫预处理。预处理过程实质上是处理“#”,将#include包含的头文件直接拷贝到hell.c当中;将#define定义的宏进行替换,同时将代码中没用的注释部分删除等
具体做的事儿如下:
(1)将所有的#define删除,并且展开所有的宏定义。说白了就是字符替换
(2)处理所有的条件编译指令,#ifdef #ifndef #endif等,就是带#的那些
(3)处理#include,将#include指向的文件插入到该行处
(4)删除所有注释
(5)添加行号和文件标示,这样的在调试和编译出错的时候才知道是是哪个文件的哪一行
(6)保留#pragma编译器指令,因为编译器需要使用它们。
第二步编译
由.i文件到.s文件,这个过程叫编译。编译的过程实质上是把高级语言翻译成机器语言的过程
(1)词法分析
(2)语法分析
(3)语义分析
(4)优化后生成相应的汇编代码
第三步汇编
由.s文件到.o文件,这个过程叫汇编。汇编过程实际上指把汇编语言代码翻译成目标机器指令的过程。
第四步链接
由.o文件到可执行文件,这个过程叫链接。就像刚才的hello.c它使用到了C标准库的东西“printf”,但是编译过程只是把源文件翻译成二进制而已,这个二进制还不能直接执行,这个时候就需要做一个动作,将翻译成的二进制与需要用到库绑定在一块。
gcc hello.c -o a可以生成可执行程序。gcc不带任何参数,ldd就可以看到你的可执行程序依赖的库。