预编译

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的内容替换为空,或者说,是直接删去

6define可以替代任何变量,这个操作只是预编译中,替换而已

7define的功能比函数强大,但是漏洞比函数大的多。

定义的内置宏,用于实现发现那个行出错的消息。

__LINE__当前行

__FILE____DATE__当前的时间和文件名

。。。

条件编译

1、条件编译的行为类似于C语言中的ifelse

2、条件编译是预编译指示命令,用于控制是否编译某段代码

即在预编译时去除掉一些代码,选择性的进行编译

#include的困惑

1#include的本质是将已经存在的文件内容嵌入到当前文件中

2#include的间接包含同样会产生嵌入文件内容的动作

如果间接的包含相同的头文件怎么办?会不会出错?

会出错,但是处理的方式是ifndef_headname_H_

条件编译的意义:

1、条件编译使得我们可以按不同的条件编译不同的代码段,

因而可以产生不同的目标代码

2#if#else#endif被预编译器处理;而ifelse语句被

编译器处理,必然被编译进目标代码

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函数

进行相等判断?为什么?

不可以,因为内存对齐会影响到两个大小形同,可能没用的数据也进行了比较