亲宝软件园·资讯

展开

C语言 编译过程

清风自在 流水潺潺 人气:0

一、初识编译器

编译器是一个广义的概念,真正的编译器由下面几个模块组成,真正的编译器是进行语法分析和语义分析的。

二、程序被编译的过程

如下,file.i 是中间代码,file.s 是一个汇编文件,file.o 是二进制文件。

预编译

预处理指令示例:gcc -E file.c -o file.i

编译

对预处理文件进行词法分析,语法分析和语义分析

分析结束后进行代码优化生成相应的汇编代码文件

编译指令示例:gcc -S file.i -o file.s

汇编

汇编指令示例:gcc -c file.s -o file.o

下面看一个源代码单步编译的示例:

demo.h

/*
    This is a header file.
*/
char* p = "Autumn";
 
int i = 0;

demo.c

#include "demo.h"
 
// Begin to define macro
 
#define GREETING "Hello world!"
 
#define INC(x) x++
 
// End
 
int main()
{   
    p = GREETING;
 
    INC(i);
 
    return 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 文件,如下:

	.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 文件:

这样就能运行了

三、小结

编译过程分为预处理,编译,汇编和链接四个阶段

加载全部内容

相关教程
猜你喜欢
用户评论