c语言编译和链接

发布于:2024-03-29 ⋅ 阅读:(14) ⋅ 点赞:(0)

一个.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;
}

          

本文含有隐藏内容,请 开通VIP 后查看