iOS技术文档No.22 Foundation_NSAutoreleasePool

一、背景
要想深入了解autorelease pool的原理,推荐以下两片文章即可:
Using Autorelease Pool Blocks
Objective-C Autorelease Pool 的实现原理
要想掌握上文中的要点,还是要废不少劲的。对于这种原理比较抽象,和实际开发编码没有直接关系的原理性的东西,常常是看一遍过一阵子很快就忘得了,为了加深印象,还是有必要系统性地梳理一遍,简单化地总结一下,加深一下印象。以下笔记也是基于以上两处文献进行总结的。
二、Autorelease Pool使用场景
1、降低内存使用峰值:
这一点不用多说,当你使用类似for循环这样的逻辑需要产生大量的中间变量时,Autorelease Pool无意是最佳的一种解决方案;
2、如果是对NSArray操作,如果可以的话推荐使用OC提供的以下api:

  • (void)enumerateObjectsUsingBlock:

  • (void)enumerateObjectsWithOptions:(NSEnumerationOptions)opts usingBlock:

  • (void)enumerateObjectsAtIndexes:(NSIndexSet *)s options:(NSEnumerationOptions)opts usingBlock:
    如果你debug一下源码就该知道为什么推荐使用它们了(内部封装了autoreleasepool),我们debug看一下:然后在采取以下命令跟踪string_weak_的值变化,如下:


    再点开enumerateObjectsUsingBlock的执行堆栈信息,看一下:


如果你再debug一下普通的for循环就不会有这些push和pop,既然enumerateObjectsUsingBlock内部有了autoreleasepool,为什么推荐使用它的原因就不多说了。
3、按照苹果给的文档说的,如果采取一些非cocoa创建的一些线程,将不会自动生成autoreleasepool给你,你需要手动去创建它。
三、Autorelease Pool的实现原理
1、Autoreleasepool的结构
每个Cocoa的线程都会默认标配一个Autorelease Pool,但是你也可以手动创建多个。从前面的操作中,也应该能隐约猜出来了些许,有push和pop操作,意味着每个pool的管理其实是一种类似栈结构的进栈出栈操作,当然pool的管理更复杂些,因为它可以创建多个,还可以嵌套创建删除。这种情况,普通的栈结构是无法满足这种需求的。如下的代码结构:

Pool的创建顺序:Pool 1 ---> Pool 2 ---> Pool 3,drain顺序是Pool 2 ---> Pool 1 --->Pool 3,如果要想实现这种顺序,采取FIFO做不到,普通的栈也不行。如果用链表操作可以做到,因为涉及到链表的首(Pool 2)或尾(Pool 3)插入,应该用双向链表来管理才合适。如下:

有人可能会有疑问,顺序为什么不是1、2、3,我觉得这些问题都不大,上面的顺序Push链表的复杂度为O(n),Pop的复杂度为O(1),反过来的话,
就是Push链表的复杂度为O(1),Pop的复杂度为O(n),如果纠结这个的可以去撸源码。
2、AutoreleasePoolPage的结构
上面介绍了,每个线程的Pool结构层次,其实是有多个PoolPage构成。
ARC下会对其中的对象会隐式执行autorelease操作,autorelease操作将一个指向对象实例的对象指针添加到PoolPage中。

添加的过程如下:

当当前PoolPage作用域一过,就会对从线程pool中执行pop操作,而pop操作,pop的过程,会遍历page堆栈,对指向的对象一一执行release操作,如果对象的retainCount变为0,即立即释放,如果对象的retaiCount大于0,不释放。当所有对象处理完(出栈完毕),最后完成pop操作。
四、为什么有了ARC还要Autorelease Pool?
这个问题之前我也想过,搜了下,没有感觉回答满意的,也没找到苹果的官方回答,这里只能自给妄自推断一下。提到OC的RC,首先要横向对比一下Android的GC,GC的内存回收是集中式回收(定期回收),而RC的回收是伴随整个运行时的,所以android机器有种时“卡”时“流畅”的感觉,而iOS总体比较均匀,缺乏像GC的集中式回收内存的类似机制,所以猜测Pool的产生也是弥补RC的这一不足,在RC基础上进行内存优化的一种手段。

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

推荐阅读更多精彩内容