《Effective Objective-C 2.0》 阅读笔记 3

23. 通过委托与数据源协议进行对象间通信

我们实际编码时已经经常使用到protocol的技术了(委托代理模式)

定义代理属性时,切记使用weak而非strong,避免“保留环”

@property (nonatomic, weak) id<EOCSomeDelegate> delegate;

24. 将类的实现代码分散到便于管理的几个分类中

为了避免一个实现文件太大,实现的方法太多,可以根据功能将类的实现分到不同的分类中。

EOCPerson类可以分成几个不同的实现文件:

EOCPerson+Friendship(.h/.m)
EOCPerson+Work(.h/.m)
EOCPerson+Play(.h/.m)

如果要使用分类中的方法,记得引入分类的头文件。

这样分散到分类中的好处是:

  1. 便于调试,编译后的符号表中,分类中的方法符号会出现分类的名称。
  2. 如果将私有方法放在名为Private的分类中,那很容易看到调试错误原因,并且在编写通用库供他人使用时,私有分类的头文件不公开,只能程序库自己能用。

25. 为第三方类的分类名称加前缀

分类机制常用在向无源码的类中新增功能。

将分类方法加入源类的操作是在运行期间系统加载分类时完成的。

如果分类中的方法名称与类中已有的方法名一样,分类中的方法就会覆盖原来的实现。解决办法就是给方法加前缀。

在整个应用程序中,类的每个实例都可以调用分类的方法。

26. 勿在分类中声明属性

除了"class-cotinuation分类",其他分类都无法向类中新增实例变量,它们无法把实现属性所需的实例变量合成。

当然,使用关联对象可以解决这种无法合成实例变量的问题:

#import <objc/runtime.h>

static const char *kFriendsPropertyKey = "kFriendsPropertyKey";
@implementation EOCPerson (Friendship)

- (NSArray *)friends {
    return objc_getAssociatedObject(self,kFriendsPropertyKey);
}

- (void)setFriends:(NSArray *)friends {
    objc_setAssociatedObject(self,kFriendsPropertyKey,
                            friends,OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

但是这样不理想,因为内存管理语义容易出错。万一你修改了属性的内存管理语义,还要记得在设置方法中修改关联对象所用的内存管理语义。所以不推荐这样做。

属性应该都定义在主接口里。分类的目的在于扩展功能,而非封装数据。

27. 使用"class-continuation分类"隐藏实现细节

"class-continuation分类"就是我们常写在实现文件中的这样一段代码:

@interface EOCPerson ()
//property here
@end

这样,可以将方法或者实例变量隐藏在本类中使用,而不暴露给公共接口。

如果属性在主接口中声明为只读,而类内部又要修改属性值,就可以在class-continuation分类中将其扩展为可读写。

28. 通过协议提供匿名对象

协议可以在某种程度上提供匿名类型。具体的对象类型可以淡化成遵守某协议的id类型。

使用匿名对象来隐藏类型名称。

如果具体类型不重要,重要的是对象能够响应特定方法,那么可使用匿名对象来表示。

29. 引用计数

引用计数变为0后“可能”就释放内存了,其实只是放回“可用内存池”,如果没有被覆写之前仍然可以访问,但这是很危险的,因为这样很可能出现野指针,造成程序崩溃。

弱引用避免保留环。

30. 以ARC简化引用计数

ARC是会自动执行retain、release、autorelease的,所以在ARC模式下是不可以直接调用内存管理方法的,具体如下方法:

  • retain
  • release
  • autorelease
  • dealloc

实际上,ARC不是通过OC的消息派发机制的,而是直接调用底层的C语言版本比如objc_retain。可以节省很多CPU周期。

若方法名以下列词语开头,则返回的对象归调用者所有, 即调用的代码要负责释放对象。

  • alloc
  • new
  • copy
  • mutableCopy

若方法名不以上述四个词语开头,则表示返回的对象不归调用者所有,返回的对象会自动释放。

这些规则所需要的内存管理事宜都有ARC自动处理。

ARC在运行期还可以起到优化作用, 比如在autorelease之后立马又调用retain的场景下,ARC在运行期可以检测这种多余的操作,利用全局数据结构中的一个标志位来决定是否需要真正执行autorelease和retain操作。

ARC只负责管理OC对象的内存,而CoreFoundation对象不归ARC管理,还需要开发者适当调用CFRetain/CFRelease。

31. 在dealloc方法中只释放引用并解除监听

每个对象生命周期结束后最终为系统回收,执行一次且仅一次dealloc方法。

在此方法中释放对象所拥有的所有引用,ARC会自动生成.cxx_destruct方法。

此方法还要做一件重要的事情,就是把配置的observation behavior都清理掉,比如NSNotificationCenter给此对象订阅过某种通知,那么应该在此注销。否则继续给对象发送通知的话会导致crash。

- (void)dealloc {
    CFRelease(coreFoundationObject);
    [[NSNotificationCenter defaultCenter] removeObserver:self];
}

如果非ARC模式,最后还要调用[super dealloc],ARC模式下就不需要。

对于开销较大或者系统稀缺的资源(如文件描述符、套接字、大块内存等),应该使用"清理方法"而非dealloc来释放。比如网络连接使用完毕后,调用close方法。
这样做的原因是:

  1. 避免保留稀缺资源的时间过长。
  2. 系统为了优化程序效率,不保证每个对象的dealloc方法都被执行。

在dealloc中,可以检测资源是否执行了清理操作,没有的话可以输出错误信息并执行一次清理操作。

有些方法不应该在dealloc方法里调用,比如

  • 执行异步任务的方法
  • 需要切换到特定线程执行的方法
  • 属性的存取方法

32. 编写“异常安全代码”时留意内存管理问题

虽然OC中异常只应发生在严重的错误中,但是有时候还是要编写代码来捕获异常。

在捕获异常时要管理好内存,防止泄漏。

比较下面的两种处理方法,明显后者更合适:

(1)

@try {
   EOCSomeClass *object = [[EOCSomeClass alloc] init];
   [object doSomethingThatMayThrow];
   [object release];//此处可能执行不到,造成内存泄漏
}
@catch (...) {
   NSLog(@"exception");
}

(2)

EOCSomeClass *object;
@try {
   object = [[EOCSomeClass alloc] init];
   [object doSomethingThatMayThrow];
}
@catch (...) {
   NSLog(@"exception");
}
@finally {
   [object release];//此处总会执行到。
}

以上是非ARC模式下的做法,如果是ARC模式也这样try/catch就有很大问题了,因为ARC针对这种情况不会自动处理release,这样做的代价很大。但是ARC还是可以生成安全处理异常所用的代码,只需要打开-fobjc-arc-exceptions编译器标志。

所以,总的来说,如果非ARC模式下必须捕获异常,那就设法保证代码能把对象清理干净; 如果是ARC下必须捕获异常,就要打开-fobjc-arc-exceptions标志。当然,如果发现程序有大量异常捕获操作时,说明你的代码需要重构了。

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

推荐阅读更多精彩内容