【C语言复习】程序的编译与链接

写在前面

程序的编译与链接是C语言中非常重要的一节。关键点在于详解C语言的程序编译和链接、宏的定义和与函数的区别、条件编译等知识。

程序的编译与链接

编译的过程

程序编译环境

在ANSI C的任何一种实现中,存在两个不同的环境。

  • 翻译环境,在这个环境中源代码被转换为可执行的机器指令。
  • 执行环境,用于实际执行代码。

程序执行过程

程序执行的过程:

  1. 程序首先要载入内存,再有操作系统的环境中,一般有操作系统完成。在独立的环境中,程序载入内存由人工安排,也可能是通过可执行代码置入只读内存中来完成。
  2. 程序的执行开始后,调用main函数。
  3. 开始执行程序代码,程序会使用一个运行时堆栈,存储函数的局部变量和返回地址。程序同时也可以使用静态内存,存储在静态内存中的变量在程序的整个执行过程中一直保存他们的值。
  4. 运行完代码,终止程序。可能是正常终止,也可能是意外终止。
  • 组成一个程序的每个源文件通过编译过程分别转换成目标代码。
  • 每个目标文件由链接器捆绑在一起,形成完整单一的可执行程序。
  • 链接器会引入标准C库中的所有用到的函数,也会搜索本地会用到的库,将程序所需要的函数也链接进入程序中。

编译链接的过程

test.c预编译阶段编译阶段汇编链接
以windows为例生成.i文件,执行预处理指令生成.s文件,负责语法、词法、语义分析和符号汇总生成可重定位目标文件.o文件,形成符号表,经过汇编指令到二进制指令到生成test.o的过程负责合并段表并进行符号表合并和符号表的重定向

如果我们在linux系统中编译.c文件,使用到的指令是:

gcc -o test.c test

如果我们想把这一个指令拆分成上面的阶段,让其分阶段进行,我们可以这样做:

gcc -E test.c -o test.i//预编译,生成.i文件
gcc -S test.c//编译,生成.s文件
gcc -c test.c//汇编,生成.o文件

预处理

预处理符号

在语言中一般内置一些预定义符号,如:

__FILE__ //进行编译的源文件
__LINE__ //文件当前的行号
__DATE__ //文件被编译的日期
__TIME__ //文件被编译的时间
__STDC__ //如果编译器遵循ANSI C,值为1,否则未定义

#define

#define可以用来定义标识符,语法:

#define NAME stuff
//如:
#define MAX 100 
#define ZS zhangsan //为张三起简名
#define DO_CYCLICALLY for(;;) //用更容易理解的符号来替换一种实现。
//如果定义的stuff太长,可以分几行写,每行的后面都加上反斜杠。反斜杠的作用是续行。

#define可以定义宏,因为define机制规定:允许把参数替换到文本中。语法:

#define name(parament-list) stuff
//parament-list 是由逗号隔开的符号表,存放可能出现在stuff中的符号
//parament-list的左括号必须与name相邻。

注意,定义宏的时候,所有用于对数值表达式进行求值的宏都要加上括号,每个参数都要加,整体也要加括号,用来避免使用宏的时候由于参数的操作符或者外界的操作符优先级问题出现错误。

#define的替换规则:

  1. 在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,首先被替换。
  2. 替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。
  3. 最后,在此对结果文件进行扫描,看看是否包含任何由#define定义的符号。如果是,重复上面操作。

注意事项:

  1. 宏参数和#define定义中可以出现其他#define 定义的符号,但是对于宏,不能进行递归。

  2. 当预处理器搜索#define 定义的符号时,字符串常量的内容并不会被搜索。

使用技巧:

  1. 妙用宏定义函数
#define PRINT(FORMAT, VALUE)\
	printf("the value is "FORMAT"\n", VALUE);
	
	
	int main()
	{

		...
		PRINT("%d", 10);
	}

说明:只有当字符串作为宏参数的时候才可以把字符串放在字符串中。

  1. 使用#, 把宏参数变成对应的字符串。

    int i = 10;
    #define PRINT(FORMAT, VALUE)\
    	printf("the value of "#VALUE" is "FORMAT "\n", VALUE);
    	...
    	PRINT("%d", i+3);
    

    代码中的#VALUE在预处理阶段会被处理为:“VALUE”。最终的输出结果时:

    the value of i+3 is 13
    
  2. 使用##,把位于##两边的符号合成一个新的符号。

    它允许宏定义从分离的文本片段创建标识符。

    #define ADD_TO_SUM(num, value)\
    	sum##num += value;
    	...
    	ADD_TO_SUM(5,10);
    

    注意:这样的链接必须产生一个合法的标识符,否则结果是未定义的。

  3. 有的宏参数是带副作用的,可能会对自己或者其他的值产生影响。如果这个宏参数在宏的定义中出现超过一次,就会产生危险。副作用的意思是:表达是在求值时出现了永久性效果。举个例子:

    x + 1;
    x ++;
    

    其中x + 1 无副作用,因为并未改变x的值;但是x++改变了x的值,所以如果在宏定义中出现多次,就会影响x的正确取值,不注意会出现错误。

宏和函数的对比:

宏通常被应用在执行简单的运算。比如两个数中找较大。不用函数的原因是:

  1. 宏比函数在程序的规模和速度方面更胜一筹。
  2. 宏是无关类型的。而函数声明则必须声明为特定的类型(只针对C语言。)

宏的缺点:

  1. 每次使用宏,一份宏定义的代码会被插入到程序中去,所以宏一般只能定义短的代码,否则会大幅度增加程序的长度。
  2. 宏无法调试。无法递归。
  3. 宏正因为类型无关,所以不够严谨。
  4. 宏可能会带来运算符优先级的问题,从而导致程序错误。

命名约定:宏名全部大写,而函数名不要全部大写。

移除宏定义:

#undef NAME

条件编译

使用条件编译,在编译一个程序的时候,将一条语句编译或者放弃编译是很方便的。
比如:调试性的代码,可以选择性的编译。

#define __DEBUG__
int main()
{
	int i = 0;
	...
	
	#ifdef __DEBUG__
	...
	#endif __DEBUG__
	
	...
}

常见的条件编译:

  1. #if 常量表达式
    	...
    #endif
    //常量表达式由预处理器求值。
    如:
    #define __DEBUG__ 1
    #if __DEBUG__
    ...
    #endif
    
  2. //多分支条件编译
    #if 常量表达式
    ..
    #elif
    
    #else
    
    #endif
    
  3. 判断是否被定义

    #if defined(symbol)
    #ifdef symbol
    
    #if !defined(symbol)
    #ifndef symbol
    
  4. 嵌套指令

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值