C语言的编译和链接

一个.c源文件是如何经过处理变成可执行的.exe文件?

这其中经过了编译和链接两个大过程。总的来讲,就是每个源文件经过编译后生成对应地目标文件,然后所有的目标文件和所引用的标准库链接,形成了.exe文件。具体是怎样,我来讲一讲。

1.编译

a.预处理

此时.c文件进入预处理阶段,执行预处理命令。比如#define,#undef,(#ifdef,#endif),#error,#line等等。预处理结束后,将生成.i文件。

b.编译

此时.i文件进入编译阶段,此时编译器对.i文件进行语法分析,词法分析,语义分析,符号汇总,其实就是把c语言指令翻译成汇编指令,那么前三个就很好理解了,符号汇总是干什么呢?符号汇总是把文件中出现的一些函数名等一些特殊的名字记录下来,比如main。之后就生成一个.s文件。

c.汇编

此时.s文件进入汇编阶段,编译器将.s文件中的汇编指令翻译成二进制指令,生成.o文件。在这一阶段,将生成符号表,符号中的符号就是编译阶段中记下的特殊的名字,而表就是把这些名字和它的地址关联起来作出一个表。比如写了一个add函数,那么将add和add的地址联系起来,但是如果在这单独的.c文件中,只有add的声明,而没有定义,则将其地址记为0.

2.链接

链接很重要的一个地方就是合并符号表,在之前讲到,如果在这单独的.c文件中,只有add的声明,而没有定义,则将其地址记为0。而在另一个.c文件内发现了add的定义,则将add函数定义的地址替换掉这个0。然后将标准库中函数的地址也进行相同操作。

最后呢,就生成了可执行的.exe文件。

然后有几个需要注意的点

1.#include "stdio.h" 和 #include <stdio.h> 的区别

""的操作是先到本地文件中寻找,然后到标准库中寻找,而<>则是直接到标准库中寻找。所以对于本地的.h文件,使用 "" ,标准库应使用 <> 。

2.宏和函数的区别

a.宏比函数更快,宏只是一个寻找替换的过程,而调用函数,要现在栈区内开辟空间,然后开辟形参等一系列操作,所以对于简单的逻辑,宏更好。

b.宏没有参数限制,宏只是寻找替换,所以根本就不会检查参数。如下面这个宏。

#define MALLOC(num, type) (type*)malloc(num * sizeof(type))


int main()
{
	int* x = MALLOC(10, int);
	if (x == NULL)
	{
		//------
		return 0;
	}
	int i = 0;
	for (i = 0; i < 10; i++)
		x[i] = i;
	for (i = 0; i < 10; i++)
		printf("%d ", x[i]);
	return 0;
}

函数肯定写不出来这种效果,应该函数的参数是有类型限制的,没有一种参数它的类型是类型。

但是这样也说明,函数比宏更为严谨。

c.宏可能会有副作用

因为宏只是寻找替换,对于这样一个宏。

#define x(a, b) ((a > b)?(a):(b))

如果传入的a和b分别为a++,b++,这很有可能把某个++执行了两边。

d.宏不能调试,不能递归,而函数可以。

e.因为宏是寻找替换,所以每使用一个宏就会替换一次,这可能大大增加了代码量,而函数则不会。

3.5个预编译符号

__FILE__
__LINE__
__DATE__
__TIME__
__STDC__
//进行编译的源文件
//文件当前的行号
//文件被编译的日期
//文件被编译的时间
//如果编译器遵循ANSI C,其值为1,否则未定义
int main()
{
	printf("%s\n %s\n %s\n %d\n",__FILE__,__DATE__,__TIME__,__LINE__);
	return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

roseisbule

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

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

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

打赏作者

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

抵扣说明:

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

余额充值