Objective-C编译器编译源代码的一般流程是:接受源文件,然后将他们转换为能够在目标平台上执行的文件,可将该过程划分为几个阶段:Objective-C编程语言含有一个预处理器,用于在编译前处理源文件。其中有些处理操作是自动执行的,另一些处理操作是根据源文件中包含的预处理器语言元素执行的。
如上图所示,这些阶段包括词法分析、语法分析、生成代码和优化、汇编与链接,最终生成可执行的二进制文件。
预处理器语言
预处理器语言是一门不同于Objective-C的独立编程语言,预处理语言对源文件进行的转换只要包括源文件的内容、条件编译和宏展开。预处理器语言元素会在程序编译前处理源文件,但预处理器不能识别OC代码。
预处理器语言定义了预处理器指令和宏展开,预处理器指令是指由预处理器(不是编译器)执行的命令,宏指令是具有名称的一段代码。当该名称在源代码中使用时,就会被替换成代表的那段代码,预处理器语言还定义了多个操作符和关键字
指令
Objective-C源代码中的预处理器指令会使用如特的语法,使他们由预处理器处理,预处理器指令具有以下形式:
#指令名 指令参数
预处理器指令以#号开头,后面紧跟指令名,之后是相应的参数,例如:
#import "ViewController.h"
就是一条预处理器指令, 预处理器指令会将换行符用作结束符号,使用预处理器指令扩展为多行,可使用反斜杠()连接两行代码。
#define kScreenWidth [UIScreen mainScreen].bounds.size.width \ *5.0
成套的预处理器指令及其作用
1.头文件包含(
#include、#import
)
2.条件编译(#ifndef、#if 、#elif、#else、#endif、#ifdef
)
3.诊断(#line、#warning、#error
)
4#pragma
指令
头文件包含
预处理器通过两条指令
#include和#impor
启动头文件的包含功能,基本上这两条指令的作用是使预处理器获得被包含文件的文本,并将其插入到当前文件中,正因如此,它们促进了代码的重用,因为源文件可以使用外部类的接口和宏指令,而无需直接复制它们。
#include指令的语法有以下两种形式:
#include "<#header#>"
#include <<#header#>>
1.使用双引号
"<#header#>"
封装头文件的名称时,编译器会先从储存文件的目录中搜索被包含的头文件,如果没有找到,编译器会在默认目录中搜索头文件,默认目录是预先配置的用于搜索系统标准头文件的目录。
2.使用尖括号<<#header#>>
封装头文件的名称时,编译器会在默认目录中搜索被包含的头文件。
#impor指令也能包含头文件,与#include指令一样,它与#include指令的区别是,它可以确保头文件仅在源文件中被包含一次,因而能够防止递归包含
条件编译
使用条件编译指令(
#ifndef、#if 、#elif、#else、#endif、#ifdef
)可以根据条件是否成立,确定包含或不包含部分或者全部源文本。
使用#if
指令可以测试表达式的值,并根据测试结果确定是否包含部分源文件
预处理指令#if
的语法
#if isIf == 100/*条件表达式*/
//条件文本
NSLog(@"if");
#endif
#if
指令用于测试条件表达式,#if指令与#endif指令配对使用,封装根据调价包含的文本。条件表达式指整型运算,可以含有下列元素:
1.整型和字符型常量;
2.算数操作符、位操作、转换、比较和逻辑操作;
3.预处理器宏指令,在计算表达式前,宏指令会被展开;
4.用于检查宏是否被定义的已定义操作符;
5.非宏指令标识符,在求值时他们都会被赋值为0。
#elif
#elif指令
用于条件判断为否的情况下,增强了#if指令
,使你可以再两个或多个条件之间进行选择,该指令位于#if
和#endif
语句对之间,只有当#if指令
后紧跟的条件表达式的值为false时,这条#elif指令
之后的条件才会被处理。
#if isIf == 100/*条件表达式*/
//条件文本
NSLog(@"if");
#elif isIf == 90/*条件表达式*/
NSLog(@"elif");
//条件文本
#endif
#else
#else指令
增强了#if
和#elif指令
,它可以再#if和#elif表达式都为false的情况下,提供包含条件文本的机制。
#if isIf == 1/*条件表达式*/
//条件文本
NSLog(@"if");
#elif isIf == 90
NSLog(@"elif");
#else
NSLog(@"else");
#endif
如果else指令前面的条件表达式都为false,那么预处理器就会将else指令中的条件文本包含到源文件时。
#ifdef指令
当
#ifdef指令
中的宏拥有定义时,不论它拥有任何值,都会被预处理器将条件文包含到源文件中。
#ifdef kScreenWidth
NSLog(@"什么东西");
#endif
#ifndef指令
#ifndef指令
可以补充#ifdef指令,当#ifndef指令`中的宏还未定义时,不论它拥有任何值,都会被预处理器将条件文包含到源文件中。
#ifndef kkkkk
//kkkkk未定义
NSLog(@"未定义宏");
#endif
在
#if
和#elif
条件表达式中使用defined
操作符,可以获得与#ifdef
和#ifndef指令
相同的效果。
defined
操作符用于测试一个名称是否被定义为宏,它等同于#ifdef
指令
诊断
在编译程序的过程中,可使用多条预处理指令诊断问题,使用
#error指令
可以使用预处理器生成一条错误消息并中止编译过程。
#waring
指令可以生成编译时警告消息,但允许编译器继续进行编译。
使用
#line
指令可以再编译器消息中添加行号,当编译出错时,编译器会显示富含出租哦文件名称和相应行号的错误信息,从而使定位错误代码变得更加容易
宏
宏
是指有名称的代码段。当在源代码中使用某个名称时,其代表的代码段就会替换它。使用预处理宏指令可以定义常量值,还可以配合输入参数值提供类似函数的功能,可使用预处理器指令#define
定义宏,使用#undef
指令移除宏
#define kScreenWidth/*宏名*/ [UIScreen mainScreen].bounds.size.width /*代码*/
#define kNUmber 100/*值*/
#define kSum(n) 10*n //宏名后可以跟一个或多个参数
int num = kSum(5);
NSLog(@"%d",num);
// num = 50
1