环境
C语言是一种编译型语言,需要把源文件进行编译之后才能运行,它的编译过程是:
https://www.jb51.net/article/245111.htm 脚本之家关于c语言编译过程
一、预处理
二、编译
三、汇编
四、连接
如下,file.i 是中间代码,file.s 是一个汇编文件,file.o 是二进制文件:
一、预处理:
1、处理所有的注释,以空格代替
2、将所有的宏定义删除,并且展开所有的宏定义
3、处理条件编译指令 #if,#ifdef,#elif,#endif
4、处理#include,展开被包含的文件
5、保留编译器需要使用的#pargma指令
处理预处理指令示例:
gcc -E file.c -o file.i
二、编译
1、对预处理文件进行词法分析
-词法分析:分析关键字、标识符,立即数等是否合法
-语法分析:分析表达式是否遵守语法规则
-语义分析:在语法分析的基础上进一步分析表达式是否合法
分析结束后进行代码优化生成相应的汇编代码文件
编译指令示例:
gcc -c file.s -o file.o
下面看一个源代码单步编译的示例:
## demo.h
在这里插入代码片
```/*
This is a header file.
*/
char* p = "Autumn";
int i = 0;
## demo.c
/*
This is a header file.
*/
char* p = "Autumn";
int i = 0;
输入 gcc -E demo.c -o demo.i,如下:
然后就生成了 demo.i 文件,如下:
# 1 "demo.c"
# 1 "<built-in>"
# 1 "<command-line>"
# 1 "demo.c"
# 1 "demo.h" 1
# 9 "demo.h"
char* p = "Autumn";
int i = 0;
# 2 "demo.c" 2
# 11 "demo.c"
int main()
{
p = "Hello world!";
i++;
return 0;
}
可以看到注释都没有了, demo.h 文件的全局变量被复制过来,宏也被替换掉了,#开头的信息是给后续编译器使用的。
输入gcc -S demo.i -o demo.s,如下
然后就生成了 demo.o 文件,如下:
然后就生成了 demo.o 文件,如下:
.file "demo.c"
.globl p
.section .rodata
.LC0:
.string "Autumn"
.data
.align 4
.type p, @object
.size p, 4
p:
.long .LC0
.globl i
.bss
.align 4
.type i, @object
.size i, 4
i:
.zero 4
.section .rodata
.LC1:
.string "Hello world!"
.text
.globl main
.type main, @function
main:
pushl %ebp
movl %esp, %ebp
movl $.LC1, p
movl i, %eax
addl $1, %eax
movl %eax, i
movl $0, %eax
popl %ebp
ret
.size main, .-main
.ident "GCC: (Ubuntu/Linaro 4.4.4-14ubuntu5.1) 4.4.5"
.section .note.GNU-stack,"",@progbits
最后输入gcc -c demo.s -o demo.o,如下:
这样就生成了一个 .o 文件
最后链接器出场了,输入 gcc demo.o,如下:
这样就生成一个 a.out 文件: