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;
}
相关推荐
进击的小头3 小时前
行为型模式:策略模式的C语言实战指南
c语言·开发语言·策略模式
爱编码的小八嘎4 小时前
C语言对话-5.通过任何其他名字
c语言
定偶6 小时前
C语言入门指南
c语言·开发语言
的卢马飞快6 小时前
【C语言进阶】给数据一个“家”:从零开始掌握文件操作
c语言·网络·数据库
我能坚持多久6 小时前
D17—C语言结构体详解:从声明、对齐到位段应用
c语言·开发语言
willingli7 小时前
c语言经典题目 91-100
c语言
傻乐u兔7 小时前
C语音进阶————数据在内存中的存储2
c语言·开发语言·算法
二年级程序员7 小时前
自定义类型:联合体与枚举
c语言
麒qiqi8 小时前
ADC 的原理与实战
c语言·开发语言·单片机·嵌入式硬件
比昨天多敲两行9 小时前
C/C++内存管理
c语言·开发语言·jvm