Effective Objective-C 2.0 编写高质量iOS与OS X代码的52个有效方法-学习笔记(第 5 - 6 章)

重新学习《Effective Objective-C 2.0 编写高质量iOS与OS X代码的52个有效方法》

并做个学习笔记,第 5 - 6 章


目录:

第5章 内存管理

第29条:理解引用计数

第30条:以ARC简化引用计数

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

第32条:编写"异常安全代码"时留意内存管理问题

第33条:以弱引用避免保留环

第34条:以"自动释放池块"降低内存峰值

第35条:用"僵尸对象"调试内存管理问题

第36条:不要使用retainCount

第6章 块与大中枢派发

第37条:理解"块"这一概念

第38条:为常用的块类型创建typedef

第39条:用handler块降低代码分散程度

第40条:用块引用其所属对象时不要出现保留环

第41条:多用派发队列,少用同步锁

第42条:多用GCD,少用performSelector系列方法

第43条:掌握GCD及操作队列的使用时机

第44条:通过Dispatch Group机制,根据系统资源状况来执行任务

第45条:使用dispatch_once来执行只需运行一次的线程安全代码

第46条:不要使用dispatch_get_current_queue


第 5 章 内存管理

第 29 条 理解引用计数

1)引用计数机制通过可以递增递减的计数器来管理内存。对象创建好之后,其保留计数至少为1.若保留计数为正,则对象继续存活。当保留计数降为0时,对象就被销毁了。

2)在对象生命期中,其余对象通过引用来保留或释放此对象。保留与释放操作分别会递增递减保留计数。

第 30 条 以 ARC 简化引用计数

1)有 ARC 之后,程序员就无须担心内存管理问题了。使用 ARC 来编程,可省去类中的许多“样板代码”。

2)ARC 管理对象生命期的办法基本上就是:在合适的地方插入“保留”及“释放”操作。在 ARC 环境下,变量的内存管理语义可以通过修饰符指明,而原来则需要手工执行“保留”及“释放”操作。

3)由方法所返回的对象,其内存管理语义总是通过方法名来实现。ARC 将此确定为开发者必须遵守的规则。

4)ARC 只负责管理 Objective-C 对象的内存。尤其要注意:CoreFoundation 对象不归 ARC 管理,开发者必须适时调用 CFRetain/CFRelease。

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

1)在 dealloc 方法里,应该做的事情就是释放指向其他对象的引用,并取消原来订阅的“键值观测”(KVO)或NSNotivicationCenter等通知,不要做其他事情。

2)如果对象持有文件描述符等系统资源,那么应该专门编写一个方法来释放此种资源。这样的类要和其使用者约定:用完资源后必须调用 close 方法。

3)执行异步任务的方法不应在 dealloc 里调用;只能在正常状态下执行的那些方法也不应在 dealloc 里调用,因为此时对象已正在回收的状态了。

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

1)捕获异常时,一定要注意将 try 块内所创立的对象清理干净。

2)在默认情况下,ARC 不生成安全处理异常所需的清理代码。开启编译器标志后,可生成这种代码,不过会导致应用程序变大,而且会降低支行效率。

第 33 条 以弱引用避免保留环

1)将某些引用设为 weak,可避免出现“保留环”。

2)weak 引用可以自动清空,也可以不自动清空。自动清空是随着 ARC 而引入的新特性,由运行期系统来实现。在具备自动清空功能的弱引用上,可以随意读取其数据,因为这种引用不会指向已经回收过的对象。

第 34 条 以“自动释放池块”降低内存峰值

1)自动释放池排布在栈中,对象收到 autorelease 消息后,系统将其放入最顶端的池里。

2)合理运用自动释放池,可降低应用程序的内存峰值。

3)@autoreleasepool 这种新式写法能创建出更为轻便的自动释放池。

第 35 条 用“僵尸对象”调试内存管理问题

1)系统在回收对象时,可以不将其真的回收,而是把它转化为僵尸对象。通过环境变量 NSZombieEnabled 可开启此功能。

2)系统会修改对象的 isa 指针,令其指向特殊的僵尸类,从而使该对象变为僵尸对象。僵尸类能够响应所有的选择子,响应方式为:打印一条包含消息内容及其接收者的消息,然后终止应用程序。

第 36 条 不要使用 retainCount

1)对象的保留计数看似有用,实则不然,因为任何给定时间点上的“绝对你试计数”(absolute retain count)都无法反映对象生命的全貌。

2)引入 ARC 之后,retainCount 方法就正式废止了,在 ARC 下调用该方法会导致编译器报错。

第 6 章 块与大中枢派发

第 37 条 理解“块”这一概念

说明:块,即 block;大中枢派发,即 Grand Central Dispatch, GCD。

1)块是 C、C++、Objective-C 中的词法闭包。

2)块可接受参数,也可返回值。

3)块可以分配在栈或堆上,也可以是全局的。分配在栈上的块可拷贝到堆里,这样的话,就和标准的 Objective-C 对象一样,具备引用计数了。

第 38 条 为常用的块类型创建 typedef

1)以 typedef 重新定义块类型,可令块变量用起来更加简单。

2)定义新类型时应遵从现有的命名习惯,勿使其名称与别的类型相冲突。

3)不妨为同一个块签名定义多个类型别名。如果要重构的代码使用了块类型的某个别名,那么只需修改相应 typedef 中的块签名即可,无须改动其他 typedef。

第 39 条 用 handler 块降低代码分散程度

1)在创建对象时,可以使用内联的 handler 块将业务逻辑一并声明。

2)在有多个实例需要监控时,如果采用委托模式,那么经常需要根据传入的对象来切换,而若改用 handler 块来实现,则可直接将块与相关对象放在一起。

3)设计 API 时如果用到了 handler 块,那么可以增加一个参数,使调用者可通过此参数来决定应该把块安排在哪个队列上执行。

第 40 条 用块引用其所属对象时不要出现保留环

1)如果块所捕获的对象直接或间接的保留了块本身,那么就得当心保留环(循环引用)问题。

2)一定要找个适当的时机解除保留环,而不能把责任推给 API 的调用者。

第 41 条 多用派发队列,少用同步锁

说明:派发队列,即 GCD;同步锁,即 @synchronized

1)派发队列可用来表述同步语义(synchronization semantic),这种做法要比使用 @synchronized 块或 NSLock 对象更简单。

2)将同步与异步派发结合起来,可以实现与普通加锁机制一样的同步行为,而这么做却不会阻塞执行异步派发的线程。

3)使用同步队列及栅栏块(dispatch_barrier_async),可以令同步行为更加高效。

第 42 条 多用 GCD,少用 performSelector 系列方法

1)performSelector 系统方法在内存管理方面容易有疏失。它无法确定将要执行的选择子具体是什么,因而 ARC 编译器也就无法插入适当的内存管理方法。

2)performSelector 系列方法所能处理的选择子太过局限了,选择子的返回值类型及发送给方法的参数个数都受到限制。

3)如果想把任务放在另一个线程上执行,那么最好不要用 performSelector 系列方法,而是应该把任务封装到块里,然后调用 GCD 的相关方法来实现。

第 43 条 掌握 GCD 及操作队列的使用时机

说明:操作队列,即 NSOperationQueue

1)在解决多线程与任务管理问题时,派发队列并非唯一文案。

2)操作队列提供了一套高层的 Objective-C API,能实现纯 GCD 所具备的绝大部分功能,而且还能完成一些更为复杂的操作,那些操作若改用 GCD 来实现,则需另外编写代码。

第 44 条 通过 Dispatch Group 机制,根据系统资源状况来执行任务

1)一系列任务可归入一个 dispatch group 之中。开发者可以在这组任务执行完毕时获得通知。

2)通过 dispatch group,可以在并发式派发队列里同时执行多项任务。此时 GCD 会根据系统资源状况来调度这些并发执行的任务。开发者若自己来实现此功能,则需编写大量代码。

第 45 条 使用 dispatch_one 来执行只需运行一次的线程安全代码

1)经常需要编写“只需执行一次的线程安全代码”。通过 GCD 所提供的 dispatch_once 函数,很容易就能实现此功能。

2)标记应该声明在 static 或 global 作用域中,这样的话,在把只需执行一次的块传给 dispatch_once 函数时,传进去的标记也是相同的。

第 46 条 不要使用 dispatch_get_current_queue

1)dispatch_get_current_queue 函数的行为常常与开发者所预期的不同。此函数已经废弃,只应做调试之用。

2)由于派发队列是按层级来组织的,所以无法单用某个队列对象来描述“当前队列”这一概念。

3)dispatch_get_current_queue 函数用于解决由不可重入的代码所引发的死锁,然而能用此函数解决的问题,通常也能改用“队列特定数据”来解决。



未完待续,传送门:

第 1 - 2 章

第 3 - 4 章

第 5 - 6 章

第 7 章

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

推荐阅读更多精彩内容