mirror of
https://github.com/NohamR/knowledge-kit.git
synced 2026-05-24 20:00:37 +00:00
86 lines
2.5 KiB
Markdown
86 lines
2.5 KiB
Markdown
# NSRange 设计之美
|
||
|
||
|
||
|
||
> typedef struct _NSRange {
|
||
NSUInteger location;
|
||
NSUInteger length;
|
||
} NSRange;
|
||
|
||
1、看到官方文档的源代码就知道 NSRange 是个结构体,但是如果是你设计一个这样的数据类型你会怎么办??
|
||
|
||
设计成结构体,然后有些属性怎么办?比如为了开发者方便,让你设计出一个办法,让开发者可以很快知道这个结构体的上限是什么?
|
||
|
||
苹果就很机智,设计了一个内联函数
|
||
|
||
```
|
||
NS_INLINE NSUInteger NSMaxRange(NSRange range) {
|
||
return (range.location + range.length);
|
||
}
|
||
```
|
||
|
||
2、什么是内联函数?
|
||
|
||
```
|
||
NS_INLINE 返回值类型 函数名(参数列表) {
|
||
//函数实现
|
||
//return ;
|
||
}
|
||
```
|
||
|
||
3、内联函数的应用
|
||
|
||
比如自定义一个弹窗
|
||
|
||
```
|
||
NS_INLINE void tipWithMessage(NSString *message){
|
||
|
||
dispatch_async(dispatch_get_main_queue(), ^{
|
||
|
||
UIAlertView *alerView = [[UIAlertView alloc] initWithTitle:@"提示" message:message delegate:nil cancelButtonTitle:nil otherButtonTitles:nil, nil];
|
||
|
||
[alerView show];
|
||
|
||
[alerView performSelector:@selector(dismissWithClickedButtonIndex:animated:) withObject:@[@0, @1] afterDelay:0.9];
|
||
|
||
});
|
||
|
||
}
|
||
|
||
```
|
||
|
||
|
||
4、内联函数的注意事项
|
||
|
||
内联函数是以代码膨胀为代价, 仅仅省去了函数调用的开销,从而提高函数的执行效率。如果执行函数内代码的时间相比于函数调用的开销大,那么效率收获会很小。
|
||
此外每一处调用内联函数的地方都会复制一遍代码,所以会使得程序的体积变大,消耗更多的代码段区域。
|
||
所以下面的情况不适合使用内联函数:
|
||
|
||
* 如果函数体内的代码比较长,使用内联将导致内存消耗比较大
|
||
* 如果函数体内出现循环,那么执行函数体内代码的时间要比函数的调用开销大
|
||
|
||
|
||
5、FOUNDATION_EXPORT
|
||
|
||
查看 NSRange 的代码还会发现一个关键词 **FOUNDATION_EXPORT**,它可以用作定义常量。
|
||
|
||
|
||
FOUNDATION_EXPORT 和 #define 都可用来定义常量。
|
||
用法
|
||
|
||
```
|
||
//.h
|
||
FOUNDATION_EXPORT NSString *const NickName;
|
||
|
||
//.m
|
||
NSString *const NickName = @"杭城小刘";
|
||
```
|
||
|
||
那么它和 **#define** 有何区别?
|
||
|
||
FOUNDATION_EXPORT 在检测字符串的值是否相等的时候效率更高
|
||
使用** NickName == MyName** 来判断,而 #define 是用 **[NickName isEqualToString:MyName]** 来判断。
|
||
|
||
|
||
* 本质上 FOUNDATION_EXPORT 是比较指针的自己
|
||
* \#define 是比较每个字符串是否相等 |