编译+链接
翻译环境,在这个环境中源代码被转换为可执行的机器指令
编译本身分为三个阶段
预编译(预处理)——>编译——>汇编
1. 在预编译阶段:
预处理选项 gcc test.c -E -o test.i
预处理完成之后就停下来,预处理之后产生的结果都放在test.i文件中
特别注意:
①头文件的包含(#include)
②define定义符号的替换和删除定义的符号
③注释删除
2. 在编译阶段:
编译选项 gcc test.c -S
编译完成之后就停下来,结果保存在test.s中
特别注意:
①根据语法分析、词法分析和词义分析把C语言代码转换成汇编代码
②符号汇总:把文件中的全局变量汇总
3. 在汇编阶段:
汇编选项 gcc test.c -c
汇编完成之后就停下来,结果保存在test.o中
特别注意:
①将编译产生的汇编代码转换成二进制指令
②形成符号表
预处理
1.预定义符号
这些预定义符号都是语言内置的。
__FILE__ //进行编译的源文件
__LINE__ //文件当前的行号
__DATE__ //文件被编译的日期
__TIME__ //文件被编译的时间
__STDC__ //如果编译器遵循ANSI C,其值为1,否则未定义
举个栗子:printf("file:%s line:%d\n", __FILE__, __LINE__);
2.#define
2.1#define 定义标识符
在define定义标识符的时候,不要在最后加上 ;
2.2 #define 定义宏
观察下面的代码段:
#define SQUARE( x ) x * x
int a = 5;
printf("%d\n" ,SQUARE( a + 1) );
你可能觉得这段代码将打印36这个值。
事实上,它将打印11. 为什么? 替换文本时,参数x被替换成a + 1,
所以这条语句实际上变成了:
printf ("%d\n",a + 1 * a + 1 );
提示:
所以用于对数值表达式进行求值的宏定义都应该用这种方式加上括号,避免在使用宏时由于参数中 的操作符或邻近操作符之间不可预料的相互作用。
2.3 #define 替换规则
在程序中扩展#define定义符号和宏时,需要涉及几个步骤。
①在调用宏时,首先对参数进行检查,看看是否包含任何由#define定义的符号。如果是,它们首先被替换。
②替换文本随后被插入到程序中原来文本的位置。对于宏,参数名被他们的值替换。
③最后,再次对结果文件进行扫描,看看它是否包含任何由#define定义的符号。如果是,就重复上 述处理过程。
注意:
①宏参数和#define 定义中可以出现其他#define定义的变量。但是对于宏,不能出现递归。
②当预处理器搜索#define定义的符号的时候,字符串常量的内容并不被搜索。
2.4带副作用的宏参数
当宏参数在宏的定义中出现超过一次的时候,如果参数带有副作用,那么你在使用这个宏的时候就可能 出现危险,导致不可预测的后果。
副作用就是表达式求值的时候出现的永久性效果。
例如:x+1;//不带副作用 x++;//带有副作用
#define MAX(a, b) ( (a) > (b) ? (a) : (b) )
int x = 5; int y = 8;
int z = MAX(x++, y++);
printf("x=%d y=%d z=%d\n", x, y, z);
//输出的结果是什么?
z = ( (x++) > (y++) ? (x++) : (y++));
x=6 y=10 z=9
结语
如果对您有帮助的话,
不要忘记点赞+关注哦,蟹蟹
如果对您有帮助的话,
不要忘记点赞+关注哦,蟹蟹
如果对您有帮助的话,
不要忘记点赞+关注哦,蟹蟹