对于C语言我们非常的熟悉,但是我们对计算机内部到底对代码进行解析和执行还是不了解,下面我就该问题用实际代码和编译来探究。对源代码的编译、链接过程;函数调用和内存指针变化。
对源代码的预处理和编译:
源代码的头文件(“#include <stdio.h>#include <assert.h>#include <malloc.h>)
#include <stdio.h>
#include<stdlib.h>
#include<assert.h>
int main()
{
void* p=malloc(500);
assert(p!=NULL);
int i=10;
printf("It's age is %d!\n",i);
free(p);
return 0;
}
通过GCC进行编译
预处理:通过命令—gcc -E test.c -o test.i,将会生成一个完整的源程序文件 test.i
注意:生成的目标文件test.i是可以打开查看的,里面以“#”开头的源文件被进行了处理,而其他代码还是没有进行处理,说明预处理阶段只是对定义的头文件和一些宏定义等等的处理,源程序还没有被处理。
通过命令gcc -S test.i -o test.s,进行源代码的编译阶段,会生成一个test.s汇编源程序文件
注意:通过记事本查看test.s文件时,里面全是汇编指令,在这一步将源程序代码通过汇编生成了汇编语言源程序文件
汇编:通过命令gcc -c test.s -o test.o对汇编源程序进行汇编,执行后生成的机器语言目标源文件test.o
注意:生成的test.o实际上是一个可重定位目标文件,是不可见的,即使用记事本打开也是乱码,必须通过后续的链接才能形成可执行不变文件
链接:通过命令gcc -O test.o -o test.exe,对可重定位目标文件进行链接,生成我们可以执行运行的可执行文件test.exe
注意:链接阶段是用链接器将编译器提供的标准函数库中的函数与可重定位目标文件相结合,合并后生成可执行目标文件。
源程序运行的最终结果: