免费开源的iOS开发学习平台

OC预处理:1-宏#define

在对源代码的编译过程中,需要一些机制来完成以下的一些功能:在编译时包含其他源文件、定义宏、根据条件决定编译时是否包含某些代码。要完成这些工作,就需要使用预处理程序。尽管在目前绝大多数编译器都包含了预处理程序,但通常认为它们是独立于编译器的。预处理过程通过读入源代码,检查包含预处理指令的语句和宏定义,对源代码进行相应的转换,并产生新的源代码提供给编译器。

预处理指令是以#号开头的代码行。#号必须是该行除了任何空白字符外的第一个字符。#后是指令关键字,在关键字和#号之间允许存在任意个数的空白字符。整行语句构成了一条预处理指令,该指令将在编译器进行编译之前对源代码做某些转换。预处理过程先于编译器对源代码进行处理,还会删除程序中的注释和多余的空白字符。

宏定义了一个代表特定内容的标识符。预处理过程会把源代码中出现的宏标识符替换成宏定义时的值。宏最常见的用法是定义代表某个值的全局符号。宏的第二种用法是定义带参数的宏,这样的宏可以象函数一样被调用,但它是在调用语句处展开宏,并用调用时的实际参数来代替定义中的形式参数。#define预处理指令是用来定义宏的,宏的作用范围是从宏定义的那一行开始,直到文件尾。

无参宏

定义格式:

 # define 宏名 宏体

作为一种约定,习惯上总是全部用大写字母来定义宏,这样易于把程序宏的宏标识符和一般变量标识符区别开来。另一种常见的宏标识符习惯以 k 开头,如#define kLength 20。宏定义结尾不能使用分号,因为宏是一种替换机制,是用宏体部分所有的字符串替换宏名,如果加了分号,会将分号也替换进去。

示例:

// 定义了一个符号常量,用PI这个符号代表常量3.14,在后面代码中要用到3.14的时候都可以用PI代替
#define PI 3.14
// 定义了一个符号常量,用LARGE代表(100 + 100),宏体中常量多于一个时需要用一对()括起来
#define LARGE (100 + 100)    
// 定义了一个符号常量,用WEBNAME代表了一个字符串"www.99ios.com"
#define WEBNAME "www.99ios.com"
// 定义了一个符号常量,用AND代表了 && 符号
#define AND &&    
// 在宏定义中,可以使用另一个宏定义的值
#define TWO_PI (2.0 * PI)   

有参宏

定义格式:

 #define 宏名(参数列表) 宏体

示例:

// 定义一个有一个参数的宏,求参数的平方值
#define SQUARE(a)    ((a) * (a))  // 宏体中所有的参数必须用一对()括起来 
// \ 被称为续行符,表示下一行是本行的延续。在 \ 符号所在行之后不能加任何空白字符
// 定义一个有两个参数的宏,求两个参数中较大的值,宏体可以是某些代码段,用{}括起来
#define LARGER(a, b)  ({ \
int m = a, n = b; m > n ? m : n; \
})
// 上面的代码与下方代码等价
#define LARGER(a, b) ({int m = a, n = b; m > n ? m :n;}) 

宏体中的参数要打上括号,因为宏体中传入的参数可以是一个表达式,如果不将参数打括号,那么可能遇到混合运算的时候可能会出现错误,例如下面的求平方的例子。

#define SQUARE(a)    ((a) * (a)) 
int a = 3, b = 4;
NSLog(@"square = %d\n", SQUARE(a + b)); // ((3 + 4) * (3 + 4))

参数不打括号那么计算的值会变成如下的情况:
(3 + 4 * 3 + 4)

上面的情况不是我们想要的,所以我们在使用有参数的宏时注意给参数打括号。

运算符"#"

出现在宏定义中的 # 运算符把跟在其后的参数转换成一个字符串。有时把这种用法的 # 称为字符串化运算符。

#define STRING(n) #n
NSLog(@"%s", STRING(www.99ios.com));

运行结果:

运算符"##"

##运算符用于把参数连接到一起。预处理程序把出现在##两侧的参数合并成一个符号。

#define CONNECT(a, b, c) (a##b##c)
NSLog(@"%f", CONNECT(13.6, 2, 3));

运行结果:

除非需要或者宏的用法恰好和手头的工作相关,否则很少有程序员会知道##运算符,绝大多数程序员从来没用过它。

示例代码

https://github.com/99ios/6.3.1