一编译过程
预编译-E
•处理所有的注释,以空格代替
•将所有的#define删除,并且展开所有的宏定义
•处理条件编译指令#if, #ifdef, #elif, #else, #endif
•处理#include,展开被包含的文件
•保留编译器需要使用的#pragma指令
编译-S
•对预处理文件进行一系列词法分析,语法分析和语义分析
•词法分析主要分析关键字,标示符,立即数等是否合法
•语法分析主要分析表达式是否遵循语法规则
•语义分析在语法分析的基础上进一步分析表达式是否合法
•分析结束后进行代码优化生成相应的汇编代码文件
汇编-C
•汇编器将汇编代码转变为机器可以执行的指令
静态链接在编译期完成,动态链接在运行期完成
二、宏定义
宏表达式中不能出现递归定义;
宏定义之后在程序的任何地方都可以引用该宏,如果不需要在后续代码中利用该宏可通过#undef
内置宏:
__FILE__ 被编译的文件名;
__LINE__ 当前行号;
__DATE__ 编译时的日期;Jan 31 2012
__TIME__ 编译时的时间;17:01:01
__STDC__编译器是否遵循标准C规范
三:条件编译:#if…#else…#endif
四:#error和#line
#error用于生成一个编译错误消息,并停止编译,#warning用于生成编译警告,但不会停止编译,例#error message
#line用于强制指定新的行号和编译文件名,并对源程序的代码重新编号,用法#line number filename(filename可省略)
其本质是重定义__LINE__和__FILE__
四:#pragma,用法#pragma parameter
#pragma是编译器指示字,用于指示编译器完成一些特定的动作
#pragma所定义的很多指示字是编译器和操作系统特有的
#pragma在不同的编译器间是不可移植的
预处理器将忽略它不认识的#pragma指令
两个不同的编译器可能以两种不同的方式解释同一条#pragma指令
内存对齐:
不同类型的数据在内存中按照一定的规则排列;而不是顺序的一个接一个的排放,这就是对齐。
CPU对内存的读取不是连续的,而是分成块读取的,块的大小只能是1、2、4、8、16字节,当读取操作的数据未对齐,则需要两次总线周期来访问内存,因此性能会大打折扣;某些硬件平台只能从规定的地址处取某些特定类型的数据,否则抛出硬件异常
struct占用的内存大小
第一个成员起始于0偏移处
每个成员按其类型大小和指定对齐参数n中较小的一个进行对齐
• 偏移地址和成员占用大小均需对齐
• 结构体成员的对齐参数为其所有成员使用的对齐参数的最大值
结构体总长度必须为所有对齐参数的整数倍
四、#和##运算
#运算符用于在预编译期将宏参数转换为字符串
#include <stdio.h>
#define CALL(f, p) (printf("Call function %s\n", #f), f(p))
int square(int n)
{
return n * n;
}
int f(int x)
{
return x;
}
int main()
{
printf("1. %d\n", CALL(square, 4));
printf("2. %d\n", CALL(f, 10));
return 0;
}
##运算符用于在预编译期粘连两个符号
#include <stdio.h>
#define NAME(n) name##n
int main()
{
int NAME(1);
int NAME(2);
NAME(1) = 1;
NAME(2) = 2;
printf("%d\n", NAME(1));
printf("%d\n", NAME(2));
return 0;
}