【C语言进阶剖析】19.C语言中的编译过程

文章目录

一、初识编译器

二、程序被编译的过程

三、小结


一、初识编译器

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

二、程序被编译的过程

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

  • 预编译
    • 处理所有的注释,以空格代替
    • 将所有的 #define 删除,并且展开所有的宏定义
    • 处理条件编译指令 #if#ifdef#elif,  #else,  #endif
    • 处理 #include,展开被包含的文件
    • 保留编译器需要使用的 #pragma 指令

        预处理指令示例: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.s 文件,如下:

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

         这样就能运行了

三、小结

  •  编译过程分为预处理,编译,汇编和链接四个阶段
    • 预处理:处理注释以及已经以 # 开头的符号
    • 编译:进行词法分析语法分析语义分析
    • 汇编:将汇编代码翻译为机器指令的目标文件
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

清风自在 流水潺潺

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值