引用计数原理

ARC

代码编译阶段,在上下文中自动成对插入MRC下的retain和release方法,保证通过引用计数正确的管理内存(针对堆上)。


iOS中引用计数的存储方案

1、TaggedPointer下的小内存对象,直接返回指针值作为引用计数。NSString会根据长度(小于60字节)决定是否使用。但深拷贝后变为普通指针。

2、nonpointer_isa:OC2.0+64位,使用对象的isa指针的第一位标记是否使用了优化后的isa,后8位来存储引用计数(溢出后移出部分正常管理)。

3、Runtime使用一张散列hash表(SideTables)来管理,SideTable中的RefcountMap属性(还有weak表自旋锁两个属性)。注:objc_object::isTaggedPointer() 获取TAG_MASK标识位以判断是否使用了TaggedPoint。


SideTable

SideTables全局hash数组长度64,实际是StripedMap类型,里面储存了64个SideTable(为了避免资源争抢,所以不存一个表里),许多obj共用一个SideTable来存储引用计数和弱引用表相关信息。


SideTable结构体

自旋锁 spinlock_t slock  // 保证原子操作防止多线程读取问题

引用计数表 RefcountMap refcnts  // 散列表结构存储对象的持有者地址和引用计数,Zombie异常时也能定位对象地址信息。

弱引用表 weak_table_t weak_table  //保存了许多对象的,所有的weak引用,对象地址作为key,weak_entries作为值保存所有指向该对象的weak指针,dealloc时把所有weak指针设为nil,避免野指针。

注:static SideTable *tableForPointer(const void *p);  // 获取对象地址的sidetable

注:一个sidetable对应一个weaktable,一个weaktable对象中有无数个weakentry(通过对象地址作为key获取对应的),每个weakentry保存了这个对象的所有弱引用。


获取引用计数:retainCount

objc_object的rootRetainCount()方法

1、判断储存逻辑(TaggedPointer直接获取 / 优化后的isa,指针的后19位即extra_rc变量 / 散列表中获取)

2、sidetable_retainCount(),先SideTable::tableForPointer(this)获取SideTable对象,table.refcnts即引用计数的hash表

3、it != table->refcnts.end()(如果相等则引用计数返回1)根据键值对以对象为key获取引用计数的值并+1返回,所以实际计数应该为retainCount-1。

注:refcnt_result += it->second >> SIDE_TABLE_RC_SHIFT(=2); // 返回时做了向右位移两位的操作,因为前两个bit,WEAKLY_REFERENCED标识是否有weak对象;DEALLOCATING是否正在析构。


修改引用计数:retain和release

总结:二者的实现机制类似,概括讲就是通过第一层 hash 算法,找到 指针变量 所对应的 sideTable。然后再通过一层 hash 算法,找到存储 引用计数 的 size_t,然后对其进行增减操作。retainCount 不是固定的 1,SIZE_TABLE_RC_ONE 是一个宏定义,实际上是一个值为 4 的偏移量。

retain方法

底层调用_objc_rootRetain、objc_object::rootRetain()、objc_object::sidetable_retain()、_slow四步方法,其中最重要的是增加引用计数的id objc_object::sidetable_retain()虚函数,详细实现如下:

SideTable& table = SideTables()[this];          // 传入对象地址获取对应的SideTable对象。

size_t& refcntStorage = table.refcnts[this];    // 获取 引用计数 的引用

!(refcntStorage & SIDE_TABLE_RC_PINNED) // 没有越界

refcntStorage += SIDE_TABLE_RC_ONE;      // 引用计数增加(实际增加了 1UL<<2 == 4)

注1:refcntStorage后两位被weak和析构状态占领,首位标识越界,所以不是增加1。

注2:refcnts为散列表,可能存了多个引用计数以处理引用计数越界情况,retainCount方法可以证明。

uintptr_t objc_object::sidetable_retainCount()  // 引用计数总返回1+计数表,所以总不为0

{ it != table.refcnts.end()  refcnt_result += it->second >> SIDE_TABLE_RC_SHIFT; }

release方法

四步方法与retain类似,其中减少引用计数的函数uintptr_t objc_object::sidetable_release(bool performDealloc)实现总结如下:

1、内部判断是否dealloc:it == table.refcnts.end()(引用计数为表中最后一个,直接标记析构,table.refcnts[this] = SIDE_TABLE_DEALLOCATING)

2、it->second < SIDE_TABLE_DEALLOCATING(在-1前验证引用计数是否为0,如果是,标记正在析构并发送dealloc消息,否则才-1。do_dealloc=true; it->second |= SIDE_TABLE_DEALLOCATING;)

3、it->second -= SIDE_TABLE_RC_ONE(引用计数-1,实际偏移两位)

注:SIDE_TABLE_DEALLOCATING作为引用计数归0的判断,减少了标记变量内存的额外占用,也避免负数产生。注:为什么isa中的extra_rc、sidetable中的refcnts中,保存的值都是真正的引用计数-1?因为获取时是+1后返回的,保证了释放时-1不会出现负数。

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