下载地址:Effective Objective-C 2.0 编写高质量iOS与OS X代码的52个有效方法
系统框架
47. 熟悉系统框架
将一系列代码封装为动态库, 并在其中放入描述其接口的头文件, 这样做出来的东西就叫框架. iOS应用不允许其中包含非系统的动态库, 所以一般第三方框架使用的都是静态库.
除了Foundation与CoreFoundation外, 还有许多系统库, 比如
- CFNetwork
- CoreAudio
- AVFoundation
- CoreData
- CoreText
要点
- 许多系统框架都可以直接使用. 其中最重要的是Foundation与CoreFoundation, 这两个框架提供了构建应用程序所需的许多核心功能
- 很多常见任务都能用框架来做, 例如音频与视频处理, 网络通信, 数据管理等.
- 请记住: 用纯C写成的框架与用OC写成的一样重要, 若想成为优秀的Objective-C开发者, 应该掌握C语言的核心概念
48. 多使用块枚举, 少用for循环
- 遍历collection有四种方式 ( for forin NSEnumerator 块枚举法) . 最基本的办法是for循环, 其次是NSEnumerator 遍历法及快速遍历法, 最新最先进的方式则是"块枚举法".
- "块枚举法"本身就能通过GCD来并发执行遍历操作, 无需另行编写代码.
- 若是提前知道待遍历的collection含有何种对象, 则应该修改块签名, 指出对象的具体类型
49. 对自定义其内存管理语义的collection使用无缝桥接
NSArray *anNSArray = @[@1, @2, @3, @4, @5];
CFArrayRef aCFArray = (__bridge CFArrayRef)anNSArray;
NSLog(@"Size of array = %li", CFArrayGetCount(aCFArray));
// output: Size of array = 5
转换操作中的__bridge告诉ARC 如何处理转换所涉及的Objective-C对象. __bridge本身的意思是: ARC仍然具备这个Objective-C对象的所有权. 而__bridge_retained则与之相反, 意味着ARC将交出对象的所有权. 若是前面那段代码改用它来实现, 那么用完数组之后就要加上CFRelease (aCFArray) 以释放其内存. 与之相似的, 反向转换可以通过__bridge_transfer来实现. 比如说, 把CFArrayRef转换为NSArray*, 并且想令ARC获得对象所有权, 那么就可以采用此种转换方式. 这三种转换方式称为"桥式转换".
通过无缝桥接技术, 可以再Foundation框架中的Objective-C对象与CoreFoundation框架中的C语言数据结构之间来回转换, 而在CoreFoundation层面创建collection时, 可以指定许多回调函数, 这些函数表示此collection应如何处理其元素. 然后, 可运用无缝桥接技术, 将其转换成具备特殊内存管理语义的Objective-C collection.
50. 构建缓存时选用NSCache而非NSDictionary
- 实现缓存时选用NSCache而非NSDictionary对象. 因为NSCache可以提供优雅的自动删减功能, 而且是 "线程安全" 的, 此外, 他与字典不同, 并不会拷贝键.
- 可以给NSCache对象设置上限, 用以限制缓存中的对象总个数及"总成本", 而这些尺度则定义了缓存删减其中对象的时机. 但是绝对不要把这些尺度当成可靠的 "硬限制", 他们仅对NSCache起指导作用
- 将NSPurgeableData与NSCache搭配使用, 可实现自动清除数据的功能, 也就是说, 当NSPurgeableData对象所占内存为系统所丢弃时, 该对象自身也会从缓存中移除.
- 如果缓存使用得当, 那么应用程序的响应速度就能提高. 只有那种重新计算起来很费事的数据, 才值得放入缓存, 比如那些需要从网络获取或从磁盘读取的数据.
51. 精简initialize与load的实现代码
- 在加载阶段, 如果类实现了load方法, 那么系统就会调用它, 分类里也可以定义此方法, 类的load方法要比分类中的先调用. 与其他方法不同, load方法不参与覆写机制.
- 首次使用某个类之前, 系统会向其发送initialize消息. 由于此方法遵从普通的覆写规则, 所以通常应该在里面判断当前要初始化的是哪个类.
- load与initialize方法都应该实现的精简一些, 这有助于保持应用程序的响应能力, 也能减少引入''依赖环" 的概率
- 无法再编译期与设定的全局变量, 可以放在initialize方法里初始化.
52 别忘了NSTimer会保留其目标对象
NSTimer会保留其参数中引用的对象, 若该对象又引用了定时器, 则会造成循环引用.
@interface NSTimer (BNBlockSupport)
+ (NSTimer *)bn_scheduledTimerWithTimeInterval:(NSTimeInterval)interval
block:(void(^)(void))block
repeats:(BOOL)repeats;
@end
@implementation NSTimer (BNBlockSupport)
+ (NSTimer *)bn_scheduledTimerWithTimeInterval:(NSTimeInterval)interval block:(void (^)(void))block repeats:(BOOL)repeats {
return [self scheduledTimerWithTimeInterval:interval target:self selector:@selector(blockInvoke:) userInfo:[block copy] repeats:YES];
}
+ (void)blockInvoke:(NSTimer *)timer {
void (^block)(void) = timer.userInfo;
if (block) {
block();
}
}
@end
这样做依然产生了循环引用, 但是因为循环引用的事NSTimer类对象, 这本身就是一个单例对象, 所以没有关系. 只需要在使用这个NSTimer对象以后, 如期调用他的invoke方法即可.