1.预处理指令
1)C语言在对源程序进行编译之前,会先对一些特殊的预处理指令作解释。预处理过程扫描源代码,对其进行初步的转换,产生新的源代码提供给编译器。
2)预处理指令是以#号开头的代码行。#号必须是该行除了任何空白字符外的第一个字符。#后是指令关键字,在关键字和#号之间允许存在任意个数的空白字符。整行语句构成了一条预处理指令,该指令将在编译器进行编译之前对源代码做某些转换。
3)预处理指令可以放在文件任何位置,他的作用范围是从它出现的位置到文件尾,习惯上我们常把它放在源程序头部,这样它的作用范围就是整个源程序文件了。
4)预处理指令可分成三类:宏定义、文件包含、条件编译
2.宏定义
1)宏定义了一个代表特定内容的标识符。预处理过程会把源代码中出现的宏标识符替换成宏定义时的值。
-
2)#define指令: 不带参数的宏定义(宏名一般大写)
- 一般形式:#define 宏名 字符串
#define COUNT 5
- 宏表示的值可以是一个常量表达式,其中允许包括前面已经定义的宏标识符。
#define ONE1 1 #define TWO2 2 #define THREE (ONE+TWO)
注意上面的宏定义使用了括号。尽管它们并不是必须的。但出于谨慎考虑,还是应该加上括号的。
NSInteger six = 0; six = THREE * TWO; //预处理过程把上面的一行代码转换成: six = (ONE+TWO) * TWO; //如果没有那个括号,就转换成 six = ONE+TWO * TWO;
- 宏还可以代表一个字符串常量
#define VERSION "Version1.0 Copyright(c)2003"
-
3)带参数的#define指令:带参数的宏定义(宏名一般大写)
- 一般形式:#define 宏名 (参数)
#define Cube (x) (x)*(x)*(x)
- 带参数的宏和函数调用看起来有些相似。可以是任何数字表达式甚至函数调用来代替参数x。这里再次提醒大家注意括号的使用。宏展开后完全包含在一对括号中,而且参数也包含在括号中,这样就保证了宏和参数的完整性。
NSInteger volume = 0; int num = 8+2; volume = Cube(num); //展开后为(8+2)*(8+2)*(8+2) //如果没有那些括号就变为8+2*8+2*8+2
带参数的宏比函数效率高,宏只会替换数值(文本对换)。不计算,不会替换双引号括住的字符串。
把可能产生副作用的操作移到宏调用的外面进行
NSInteger volume = 0; int num = 8+2; volume = Cube(num++);//不安全的用法 /**展开后是这样的: volume=(num++)*(num++)*(num++); 很显然,结果是10*11*12,而不是10*10*10; */ volume = Cube(num);//安全的用法 num++;
-
4)#运算符
- 出现在宏定义中的#运算符把跟在其后的参数转换成一个字符串。有时把这种用法的#称为字符串化运算符。
#define PASTE (n) "hello "#n main() { NSLog(@"%@",PASTE(2018)); //宏定义中的#运算符告诉预处理程序,把源代码中任何传递给该宏的参数转换成一个字符串。所以输出应该是hello 2018 }
-
5)##运算符
- ## 运算符用于把参数连接到一起。预处理程序把出现在##两侧的参数合并成一个符号。
#define NUM (a,b,c) a##b##c #define STR (a,b,c) a##b##c main() { NSLog("%ld ",NUM(1,2,3)); NSLog("%@ ",STR("aa","bb","cc")); //输出结果为123 aabbcc }
- 除非需要或者宏的用法恰好和手头的工作相关,否则很少有程序员会知道##运算符。绝大多数程序员从来没用过它。
3.条件编译指令
-
条件编译指令将决定那些代码被编译,而哪些是不被编译的。可以根据表达式的值或者某个特定的宏是否被定义来确定编译条件。
-
预编译指令中的表达式与C语言本身的表达式基本一至.如逻辑运算、算术运算、位运算等均可以在预编译指令中使用。
-
之所以能够实现条件编译是因为预编译指令是在编译之前进行处理的,通过预编译进行宏替换、条件选择代码段,然后生成最后的待编译代码,最后进行编译。
指令用途
#空指令,无任何效果
#include包含一个源代码文件
#define定义宏
#undef取消已定义的宏
#if如果给定条件为真,则编译下面代码
#ifdef如果宏已经定义,则编译下面代码
#ifndef如果宏没有定义,则编译下面代码
#elif如果前面的#if给定条件不为真,当前条件为真,则编译下面代码
#endif结束一个#if……#else条件编译块
#error停止编译并显示错误信息
编译方法
- #if ... #elif ... #else ... #endif
#if 条件 1
代码段 1
#elif 条件 2
代码段 2
...
#elif 条件 n
代码段 n
#else
代码段 n+1
#endif
#if的一般含义是,如果#if后面的常量表达式为true,则编译它所控制的代码.
如果条件1成立时就代码段1,条件1不成立再看条件2是否成立;如果条件2成立则编译代码段2,否则再依次类推判断其它条件;如果条件1-N都不成力则会编译最后的代码段n+1.
务必不能忘了#endif
- #ifdef ... #else ... #endif或#ifndef ... #else ... #endif
#ifdef macro_name
代码段 1
#else
代码段 2
#endif
#ifndef macro_name
代码段 1
#else
代码段 2
#endif
#ifdef指令与#ifndef指令,它们分别表示“如果有定义”及“如果无定义”。
有定义是指在编译此段代码时是否有某个宏通过 #define指令定义的宏,#ifndef指令指找不到通过#define定义的某宏
该宏可以是在当前文件此条指令的关面定义的,也可以是在其它文件中,但在此指令之前包含到该文件中的。
-
#ifdef指令与#if defined()用法一致
#if defined (宏) 代码. #endif
- 这个#if后面接的是一个宏,意思是前面的宏定义里面有没有定义这个宏,如果定义了,编译器就会编译中间的代码,如果没有定义,那就不会编译,不管这个宏定义的是什么东西,对不对等
#ifndef指令与if !defined()用法一致(取反的意思)
3、defined(macro_name)
参数为宏名(无需加""),如果该macro_name定义过则返回真,否则返回假
用该函数则可以写比较复杂的条件编译指令
#if defined(macro1) || (!defined(macro2) && defined(macro3))
...
#else
...
#endif