编写高质量代码的52个有效方法(七)---系统框架

outstanding.png

47.熟悉系统框架

框架:将一系列代码封装为动态库(dynamic library),并在其中放入描述其接口的头文件,这样做出来的东西叫做框架。

在为Mac OS X或iOS系统开发“带图形界面的应用程序”时,会用到名为Cocoa的框架,在iOS上称为Cocoa Touch。其实Cocoa本身并不是框架,但是里面集成了一批创建应用程序时经常会用到的框架。

要点:

  1. 许多系统框架都可以直接使用。其中最重要的是Foundation与CoreFoundation,这两个框架提供了构建应用程序所需要的许多核心功能;
  2. 很多常见任务都能用框架来做,例如音频与视频处理、网络通信,数据管理等;
  3. 请记住:用纯C写成的框架与ObjectiveC写成的一样重要,若想成为优秀的Objective-C开发者,应该掌握C语言的核心概念。

48.多用块枚举,少用for循环

//三个举例集合
NSArray *array = @[@1,@2,@3];
    NSDictionary *dict = @{
                           @"key1":@"value1",
                           @"key2":@"value2",
                           @"key3":@"value3"
                           };
    
NSSet *set = [[NSSet alloc] initWithObjects:@"str1",@"str2",@"str3", nil];
    
//for 循环
for (int i = 0; i < array.count; i++) {
    id object = array[i];
    //do something with object
}
    
//对于字典中间做一个allkeys的转换
NSArray *keys = [dict allKeys];
    
//对于set集合中间做一个allobjects的转换
NSArray *objects = [set allObjects];


//Objective-C 1.0的NSEnumerator来遍历
- (NSArray *)allObjects;
- (id)nextObject;


//Objective-C 1.0的NSEnumerator来遍历
NSEnumerator *enumerator = [array objectEnumerator];
id object;//若知道具体类型,可直接写成具体类型
while ((object = [enumerator nextObject]) != nil) {
    //do something with object
}
    
NSEnumerator *dicEnumerator = [dict keyEnumerator];
id key;//若知道具体类型,可直接写成具体类型
while ((key = [dicEnumerator nextObject]) != nil) {
    id value = [dict objectForKey:key];//若知道具体类型,可直接写成具体类型
    // do something with value
}
    
NSEnumerator *setEnumerator = [set objectEnumerator];
id setObject;//若知道具体类型,可直接写成具体类型
while ((setObject = [setEnumerator nextObject]) != nil) {
       // do something with object
}
    
#warning !!!若需要反向遍历,则可以使用反向枚举器
[array reverseObjectEnumerator];



//Objective-C 2.0的快速遍历  for in
    
for (id object in array) {
    // do something with object
}
    
for (id key in dict) {
    id object = [dict objectForKey:key];
    // do something with object
}
    
for (id object in set) {
    // do something with object
}


//基于块的遍历方式
[array enumerateObjectsUsingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    // do something with object    
}];
    
[dict enumerateKeysAndObjectsUsingBlock:^(id  _Nonnull key, id  _Nonnull obj, BOOL * _Nonnull stop) {
    // do something with object    
}];
    
[set enumerateObjectsUsingBlock:^(id  _Nonnull obj, BOOL * _Nonnull stop) {
    // do something with object    
}];
    
//若要反向遍历,则可使用带有“选项掩码”(option mask)的方法
[array enumerateObjectsWithOptions:NSEnumerationReverse usingBlock:^(id  _Nonnull obj, NSUInteger idx, BOOL * _Nonnull stop) {
    // do something with object    
}];

要点:

  1. 遍历collection有四种方式。最基本的办法是for循环,其次是NSEnumeratior遍历法及快速遍历法,最新、最先进的方式则是“块枚举法”。
  2. “块枚举法”本身就能通过GCD来并发执行遍历操作,无须另行编写代码。而采用其他遍历方式则无法轻易实现这一点。
  3. 若提前知道待遍历的collection含有何种对象,则应修改块签名,指出对象的具体类型。

49.对自定义其内存管理语义的collection使用无缝桥接

要点:

  1. 通过无缝桥接技术,可以在Foundation框架中的Objective-C对象与CoreFoundation框架中的C语言数据结构之间来回转换。
  2. 在CoreFoundation层面创建collection时,可以指定许多回调函数,这些函数表示此collection应如何处理其元素。然后,可运用无缝桥接技术,将其转换成具备特殊内存管理语义的Objective-C collection。

50.构建缓存时,选用NSCache而非NSDictionary

之前在阅读SDWebImage源码时,有看到里面缓存使用的就是NSCache,其使用可查看文章
关于缓存类NSCache

要点:

  1. 实现缓存时应选用NSCache而非NSDictionary对象。因为NSCache可以提供优雅的自动删减功能,而且是“线程安全的”,此外,它与字典不同,并不会拷贝键。
  2. 可以给NSCache对象设置上限,用以限制缓存中的对象总个数及“总成本”,而这些尺度则定义了缓存删减其中对象的时机。但是绝对不要把这些尺度当成可靠的“硬限制”,他们仅对NSCache起知道作业
  3. 将NSPurgeableData与NSCache搭配使用,可实现自动清除数据的功能,也就是说,当NSPurgeableData对象所占内存为系统所丢弃时,该对象自身也会从缓存中移除。
  4. 如果缓存使用得当,那么应用程序的响应速度就能提高。只有那种“重新计算起来很费事的”数据,才值得放入缓存,比如那些需要从网络获取或从磁盘读取的数据

51.精简initialize与load的实现代码

+ (void)load; //(加载进系统时调用)
+ (void)initialize;//(类在使用前的初始化调用,系统调用)

// + (void)load
对于加入运行期系统中的每个类(class)及分类(category)来说,必定会调用此方法,而且仅调用一次。当包含类或分类的程序库载入系统时,就会执行此方法,而这通常就是指应用程序启动的时候,若程序是为iOS平台设计的,则肯定会在此时执行。如果分类和其所属的类都定义了load方法,则先调用类里面的,再调用分类里的,因为分类里面可能会使用到类里面的方法。

有个重要的事情需注意,那就是load方法并不像普通的方法那样,它并不遵从那套继承规则。如果某个类本身没实现load方法,那么不管其各级超类是否实现此方法,系统都不会调用。

而且load方法务必实现的精简一些,也就是要尽量减少其所执行的操作,因为整个应用程序在执行load方法时都会阻塞。如果load方法中包含繁杂的代码,那么应用程序在执行期间就会变得无响应。不要在里面等待锁,也不用调用可能加锁的方法。

//+ (void)initialize;
对于每个类来说,该方法会在程序首次使用该类之前调用,且只调用一次。它是由运行期系统来调用的,绝不应该通过代码直接调用。

首先,它是“惰性调用的”,也就是说,只有当程序用到了相关的类时,才会调用。因此,如果某个类一直没有使用,那么其initialize方法就一直不会运行。这也就是说,应用程序无须先把每个类的initialize都执行一遍,这与load方法不同,对于load来说,应用程序必须阻塞并等着所有类的load都执行完,才能继续。

其次,此方法与load方法还有个区别,就是运行期系统在执行该方法时,是处于正常状态的,因此,从运行期系统完整度上来讲,此时可以安全使用并调用任意类中的任意方法。而且,运行期系统也能确保initialize方法一定会在“线程安全的环境”中执行,这就是说,只有执行initialize的那个线程可以操作类或类实例。其他线程都要先阻塞,等着initialize执行完。

最后,initialize方法与其他消息一样,如果某个类未实现它,而其超类实现了,那么就会运行超类的实现代码。

要点:

  1. 在加载阶段,如果类实现了load方法,那么系统就会调用它。分类里也可以定义此方法,类的load方法要比分类中的先调用。与其他方法不同,load方法不参与覆写机制。
  2. 首次使用某个类之前,系统会向其发生initialize消息。由于此方法遵从普通的覆写规则,所以通常应该在里面判断当前要初始化的是哪个类。
  3. load与initialize方法都应该实现的精简一些,这有助于保持应用程序的响应能力,也能减少引入“依赖环”的几率。
  4. 无法在编译期设定的全局常量,可以放在initialize方法里初始化。

52.别忘了NSTimer会保留其目标对象

#import <Foundation/Foundation.h>

@interface NSTimer (BlockTimer)

+ (NSTimer *)timer_scheduledTimerWithTimeInterval:(NSTimeInterval)interval block:(void(^)(void))block repeats:(BOOL)repeats;

@end



#import "NSTimer+BlockTimer.h"

@implementation NSTimer (BlockTimer)


+ (NSTimer *)timer_scheduledTimerWithTimeInterval:(NSTimeInterval)interval block:(void(^)(void))block repeats:(BOOL)repeats
{
    return [self scheduledTimerWithTimeInterval:interval target:self selector:@selector(timer_blockInvoke:) userInfo:[block copy] repeats:repeats];
}

+ (void)timer_blockInvoke:(NSTimer *)timer
{
    void (^block)(void) = timer.userInfo;
    if (block) {
        block();
    }
}

@end



//use

//本类拥有实例变量 _timer
- (void)startTimer
{
    __weak typeof(self) weakSelf = self;
    
    _myTimer = [NSTimer timer_scheduledTimerWithTimeInterval:5.0 block:^{
        ViewController *strongSelf = weakSelf;
        [strongSelf dosomething];
    } repeats:YES];
}

- (void)dosomething
{
    
}

- (void)stopTimer
{
    [_myTimer invalidate];
    _myTimer = nil;
}

- (void)dealloc
{
    [_myTimer invalidate];
}

要点:

  1. NSTimer对象会保留其目标,直到计时器本身失效为止,调用invalidate方法可令计时器失效,另外,一次性的计时器在触发任务之后也会失效。
  2. 反复执行任务的计时器,很容易引入保环,如果这种计时器的目标对象又保留了计时器本身,那肯定会导致保留环。这种环状保留关系,可能是直接发生的,也可能是通过其他对象间接发生的。
  3. 可以扩充NSTimer的功能,用“块”来打破保留环。不过,除非NSTimer将来在公共接口里提供此功能,否则必须创建分类,将相关实现代码加入其中。



PDF格式的资料来自iOS开发交流群、感觉作者的贡献,对于知识的系统归纳总结很有帮助。
编写高质量代码的52个有效方法
编写高质量代码的52个有效方法(一)—熟悉OC
编写高质量代码的52个有效方法(二)—对象、消息、运行期
编写高质量代码的52个有效方法(三)—接口与API设计
编写高质量代码的52个有效方法(四)—协议与分类
编写高质量代码的52个有效方法(五)—内存管理
编写高质量代码的52个有效方法(六)—块与大中枢派发
编写高质量代码的52个有效方法(七)---系统框架

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容