预编译
1、处理所有的注释,以空格代替
2、将所有的#define删除,并且展开所有的宏定义
3、处理条件编译指令#if,#ifdef,#elif,#else,#endif
4、处理#include,展开被包含的文件
5、保留编译器需要使用的#pragma指令
编译
1、对预处理文件进行一系列词法分析,语法分析和语义分析
2、词法分析主要分析关键字,标示符,立即数等是否合法
3、语法分析主要分析表达式是否遵循语法规则
4、语义分析在语法分析的基础上进一步分析表达式是否合法
5、分析结束后进行代码优化生成相应的汇编代码文件
汇编
1、汇编器将汇编代码转变为机器可以执行的指令
2、每个汇编语句几乎都对应一条机器指令
链接器的作用
1、连接器的主要作用是把各个模块之间相互引用的部分
2、处理好,使得各个模块之间能够正确的衔接。
编译器将编译工作主要分为预处理,编译和汇编三部
1、连接器的工作是把各个独立的模块链接为可执行程序
2、静态链接在编译期完成,动态链接在运行期完成
#define
1、#define表达式给有函数调用的假象,却不是函数
2、#define表达式可以比函数更强大
3、#define表达式比函数更容易出错
4、最好不要定义时间点的量,如i++,
5、#definePT允许这样定义,但没有仍和意义,将PT的内容替换为空,或者说,是直接删去
6、define可以替代任何变量,这个操作只是预编译中,替换而已
7、define的功能比函数强大,但是漏洞比函数大的多。
定义的内置宏,用于实现发现那个行出错的消息。
__LINE__当前行
__FILE____DATE__当前的时间和文件名
。。。
条件编译
1、条件编译的行为类似于C语言中的if…else
2、条件编译是预编译指示命令,用于控制是否编译某段代码
即在预编译时去除掉一些代码,选择性的进行编译
#include的困惑
1、#include的本质是将已经存在的文件内容嵌入到当前文件中
2、#include的间接包含同样会产生嵌入文件内容的动作
如果间接的包含相同的头文件怎么办?会不会出错?
会出错,但是处理的方式是ifndef_headname_H_
条件编译的意义:
1、条件编译使得我们可以按不同的条件编译不同的代码段,
因而可以产生不同的目标代码
2、#if…#else…#endif被预编译器处理;而if…else语句被
编译器处理,必然被编译进目标代码
3、实际工程中条件编译主要用于一下两种情况:
不同的生产线共用一份代码
区分编译产品的调试版和发布版
通过编译器命令行能够定义预处理器使用的宏
条件编译可以避免重复包含头同一个头文件
条件编译是在工程开发中可以区别不同产品线的代码
条件编译可以定义产品的发布版和调试版
#error用于生成一个编译错误消息,并停止编译
用法
#errormessage
注:message不需要用双引号包围
#error编译指示字用于自定义程序员特有的编译错误消息
类似的,
#warning用于生成编译警告,但不会停止编译
#line用于强制指定新的行号和编译文件名,并对源程序
的代码重新编号
用法
#linenumberfilename
注:filename可省略
#line编译指示字的本质是重定义__LINE__和__FILE__
用途:无关表示部分的代码有某个人开发的,开发的那个程序段出现错误,
也就是告诉调试者的写的代码的相对位置进行检错。
现在有个更好的处理技术,来处理这个问题的。
#pragma简介
1、#pragma是编译器指示字,用于指示编译器完成一些特定的动作
2、#pragma所定义的很多指示字是编译器和操作系统特有的
3、#pragma在不同的编译器间是不可移植的
4、预处理器将忽略它不认识的#pragma指令
5、两个不同的编译器可能以两种不同的方式解释同一条#pragma指令
6、确定编程的的版本,便于差错,用于显示自己的程序的版本
提示作者版本是否正确
#pragmamessage("CompileAndroidSDK2.0...")
深入:
1、预处理器将忽略它不认识的#pragma指令
2、两个不同的编译器可能以两种不同的方式解释同一条#pragma指令
一般用法:
#pragmaparameter
注:不同的parameter参数语法和意义各不相同
#pragma的内存对齐
构体变量是否可以直接用memcmp函数
进行相等判断?为什么?
不可以,因为内存对齐会影响到两个大小形同,可能没用的数据也进行了比较