详解C语言之预处理(上)
程序的翻译环境
源文件被转换成可执行的机器指令时所处的环境称为翻译环境。
由源文件(.c)转换成可执行文件(.exe)需要两步
编译通过编译器实现,链接通过链接器实现
每个源文件都会经过编译器处理后生成对应的目标文件,然后链接器将目标文件和链接库链接在一起生成可执行程序
编译和链接的具体操作
编译
编译分为预编译、编译和汇编
预编译:
1.#include<>头文件的包含,即将头文件的引用替换为函数具体的声明。
2.删除注释
3.#define,预处理操作,将define定义的替换为实际值
编译:
将C语言代码翻译为汇编代码
语法分析,词法分析,语义分析,符号汇总(函数名,全局变量)
汇编:
将汇编代码转换为二进制代码,形成符号表
链接
链接分为合并段表和符号表的合并和符号的重定位
合并段表:
目标文件都有一定的格式,分为几个段。链接器会将目标文件的相同的段里的数据合并到一起。
符号表的合并和符号的重定位:
链接器会将符号表合并为一张表,合并后当符号有冲突时,无效地址将被重新定位为有效地址,即合并后
链接操作完成后可执行程序就生成了
#define的用法
1.#define定义标识符,例如
define定义标识符时最好不要加分号";" 否则容易导致语法错误
2.#define定义宏
宏的申明方式:
#define name(parament-list) stuff
其中parament-list是由逗号隔开的符号表,可能出现在stuff中
注意:参数的左括号必须与name相邻
下面通过一些代码来看看使用宏时需要注意的问题
输出结果为11,而非36。问题出在哪呢,我们要明确函数和宏的区别,函数是传递参数的值,而宏是替换参数。#define的实质就是替换。
所以X会替换成表达式5 + 1,5+1*5+1结果显然为11。如果要改进的话在使用宏时可以多加括号,不要吝啬括号
再看另一个例子
我们发现结果依然不是我们想象那样为100,而是55。要知道不仅参数是替换,整个式子也是替换的。则DOUBLE(5)会替换成 (5) + ( 5),那么10*(5)+(5)结果为55,如果要改进的话则依然是加括号
所以记住一点,使用宏的时候要不要吝啬括号。
3.#define实现将参数插入到字符串中
实现如下 #会将X变成字符串"X"
输出结果:
hello aworld