宏定义
在了解宏定义的使用前,需要说明一下合理使用宏定义确实可以提高程序的可读性,但是滥用宏定义,也会带来一些害处。这里对宏定义使用的优缺点做个简单的说明:
优点
-
提高了程序的可读性,同时也方便进行修改,用户只需要在一处定义,多处使用,修改也只需要修改一处
-
提高程序的运行效率:使用带参的宏定义既可完成函数调用的功能,又能避免函数的出栈与入栈操作,减少系统开销,提高运行效率,如果有一个函数会在工程中频繁使用,可以考虑一下宏定义
缺点
-
由于是直接嵌入的,所以代码可能相对多一点
-
嵌套定义过多可能会影响程序的可读性,而且很容易出错;
-
对带参的宏而言,由于是直接替换,并不会检查参数是否合法,存在安全隐患。
- 说白了,预编译语句仅仅是简单的值代替,缺乏类型的检测机制
宏
宏可以分为两类,一类称为对象宏,一类称为函数宏
对象宏
对象宏可以定义一个量,如圆周率:
#define PI 3.1415926
通过宏可以拿到这个变量:
float a = PI;
NSLog(@" pi float = %f",a);
int b = PI;
NSLog(@" pi int = %d",b);
打印结果:
2019-03-10 10:30:04.824899+0800 MyObjC[65078:9988537] pi float = 3.141593
2019-03-10 10:30:04.825080+0800 MyObjC[65078:9988537] pi int = 3
但是看一下另外一种情况:
#define width 30+40
int c = width;
NSLog(@" width int = %d",c);
int d = width * 2;
NSLog(@" width int = %d",d);
结果如下:
2019-03-10 10:40:04.571089+0800 MyObjC[65252:10005979] width float = 70
2019-03-10 10:40:04.571200+0800 MyObjC[65252:10005979] width float = 110
为什么*2的结果不是140,是110呢?
因为前面提到过了,宏只是简单地替换,所以2的时候,实际是30+402
函数宏
函数宏的作用就类似于一个函数一样。如:
#define MAX(A,B) A > B ? A : B
int e = MAX(20, 40);
NSLog(@" max int = %d",e);
其结果如下:
2019-03-10 10:46:23.954484+0800 MyObjC[65351:10018292] max float = 40
再来看另外一个例子:
#define IS_LEAP_YEAR(y) y%4==0&&y%100!=0||y%400==0
if (IS_LEAP_YEAR(2019)) {
NSLog(@" 2019是润年");
}else{
NSLog(@" 2019不是润年");
}
这里需要注意一个问题,如果宏太长,可以使用"/"进行换行
#define IS_LEAP_YEAR2(y) y%4==0&&y%100!=0 \
||y%400==0
这样便是大大增加了可读性
在OC中使用字符串都需要使用@"",如果想直接使用字符串可以添加一个"#":
#define STR(x) # x
NSLog(@" str: %s",STR(aaaa));
2019-03-10 11:31:16.141783+0800 MyObjC[66841:10106049] str: aaaa
再看一个关于##的操作符,连接符##用来将两个token连接为一个token:
#define PARSER(N) printf("token" #N " = %d\n", token##N)
int token64 = 64;
int token32 = 32;
PARSER(64);//用来打印token64
PARSER(32);//用来打印token32
可变参数宏
提到可变参数宏,其实能够想到的,最多的便是NSLog。可变参数,可以利用VA__ARGS进行读取:
#define NSLog(format, ...) fprintf(stderr, "<%s : %d> %s\n", \
[[[NSString stringWithUTF8String:__FILE__] lastPathComponent] UTF8String], __LINE__, __func__); \
(NSLog)((format), ##__VA_ARGS__); \
fprintf(stderr, "-------\n"); \
mylog(@"这是自定log %d",20)
打印结果如下:
<DefineViewController.m : 58> -[DefineViewController viewDidLoad]
2019-03-10 12:49:18.940393+0800 MyObjC[68204:10216210] 这是自定log 20
-------
本文是来自我公号的文章,欢迎大家关注我的公众号,更快的更新更友爱的微信群更全的源码示例。
网友评论