runtime-闲聊内存管理

前言

ARC作为一个老生常谈的话题,基本被网上的各种博客说尽了。但是前段时间朋友通过某些手段对YYModel进行了优化,提高了大概1/3左右的效率,在观赏过他改进的源码之后我又重新看了一遍ARC相关的实现源码,主要体现ARC机制的几个方法分别是retainrelease以及dealloc,主要与strongweak两者相关

ARC的内存管理

来看看一段ARC环境下的代码
- (void)viewDidLoad {
NSArray * titles = @[@"title1", @"title2"];
}
在编译期间,代码就会变成这样:

- (void)viewDidLoad {
    NSArray * titles = @[@"title1", @"title2"];
    [titles retain];
    ///  .......
    [titles release];
}

简单来说就是ARC在代码编译阶段,会自动在代码的上下文中成对插入retain以及release,保证引用计数能够正确管理内存。如果对象不是强引用类型,那么ARC的处理也会进行相应的改变


下面会分别说明在这几个与引用计数相关的方法调用中发生了什么

retain

强引用有retainstrong以及__strong三种修饰,默认情况下,所有的类对象会自动被标识为__strong强引用对象,强引用对象会在上下文插入retain以及release调用,从runtime源码处可以下载到对应调用的源代码。在retain调用的过程中,总共涉及到了四次调用:

  • id _objc_rootRetain(id obj)
    对传入对象进行非空断言,然后调用对象的rootRetain()方法
  • id objc_object::rootRetain()
    断言非GC环境,如果对象是TaggedPointer指针,不做处理。TaggedPointer是苹果推出的一套优化方案,具体可以参考深入了解Tagged Pointer一文
  • id objc_object::sidetable_retain()
    增加引用计数,具体往下看
  • id objc_object::sidetable_retain_slow(SideTable& table)
    增加引用计数,具体往下看

在上面的几步中最重要的步骤就是最后两部的增加引用计数,在NSObject.mm中可以看到函数的实现。这里笔者剔除了部分不相关的代码:

#define SIDE_TABLE_WEAKLY_REFERENCED (1UL<<0)
#define SIDE_TABLE_DEALLOCATING      (1UL<<1)
#define SIDE_TABLE_RC_ONE            (1UL<<2)
#define SIDE_TABLE_RC_PINNED         (1UL<<(WORD_BITS-1))

typedef objc::DenseMap<DisguisedPtr<objc_object>,size_t,true> RefcountMap;
struct SideTable {
    spinlock_t slock;
    RefcountMap refcnts;
    weak_table_t weak_table;
}

id objc_object::sidetable_retain()
{
    // 获取对象的table对象
    SideTable& table = SideTables()[this];

    if (table.trylock()) {

        // 获取 引用计数的引用
        size_t& refcntStorage = table.refcnts[this];
        if (! (refcntStorage & SIDE_TABLE_RC_PINNED)) {
            // 如果引用计数未越界,则引用计数增加
            refcntStorage += SIDE_TABLE_RC_ONE;
        }
        table.unlock();
        return (id)this;
    }
    return sidetable_retain_slow(table);
}
  • SideTable这个类包含着一个自旋锁slock来防止操作时可能出现的多线程读取问题、一个弱引用表weak_table以及引用计数表refcnts。另外还提供一个方法传入对象地址来寻找对应的SideTable对象

  • RefcountMap对象通过散列表的结构存储了对象持有者的地址以及引用计数,这样一来,即便对象对应的内存出现错误,例如Zombie异常,也能定位到对象的地址信息

  • 每次retain后以后引用计数的值实际上增加了(1 << 2) == 4而不是我们所知的1,这是由于引用计数的后两位分别被弱引用以及析构状态两个标识位占领,而第一位用来表示计数是否越界。

由于引用计数可能存在越界情况(SIDE_TABLE_RC_PINNED位的值为1),因此散列表refcnts中应该存储了多个引用计数,sidetable_retainCount()函数也证明了这一点:

#define SIDE_TABLE_RC_SHIFT 2
uintptr_t objc_object::sidetable_retainCount()
{
    SideTable& table = SideTables()[this];
    size_t refcnt_result = 1;

    table.lock();
    RefcountMap::iterator it = table.refcnts.find(this);
    if (it != table.refcnts.end()) {
        refcnt_result += it->second >> SIDE_TABLE_RC_SHIFT;
    }
    table.unlock();
    return refcnt_result;
}

引用计数总是返回1 + 计数表总计这个数值,这也是为什么经常性的当对象被释放后,我们获取retainCount的值总不能为0。至于函数sidetable_retain_slow的实现和sidetable_retain几乎一样,就不再介绍了

release

release调用有着跟retain类似的四次调用,前两次调用的作用一样,因此这里只放上引用计数减少的函数代码:

uintptr_t objc_object::sidetable_release(bool performDealloc)
{
#if SUPPORT_NONPOINTER_ISA
    assert(!isa.indexed);
#endif
    SideTable& table = SideTables()[this];

    bool do_dealloc = false;

    if (table.trylock()) {
        RefcountMap::iterator it = table.refcnts.find(this);
        if (it == table.refcnts.end()) {
            do_dealloc = true;
            table.refcnts[this] = SIDE_TABLE_DEALLOCATING;
        } else if (it->second < SIDE_TABLE_DEALLOCATING) {
            // SIDE_TABLE_WEAKLY_REFERENCED may be set. Don't change it.
            do_dealloc = true;
            it->second |= SIDE_TABLE_DEALLOCATING;
        } else if (! (it->second & SIDE_TABLE_RC_PINNED)) {
            it->second -= SIDE_TABLE_RC_ONE;
        }
        table.unlock();
        if (do_dealloc  &&  performDealloc) {
            ((void(*)(objc_object *, SEL))objc_msgSend)(this, SEL_dealloc);
        }
        return do_dealloc;
    }

    return sidetable_release_slow(table, performDealloc);
}

release中决定对象是否会被dealloc有两个主要的判断

  • 如果引用计数为计数表中的最后一个,标记对象为正在析构状态,然后执行完成后发送SEL_dealloc消息释放对象
  • 即便计数表的值为零,sidetable_retainCount函数照样会返回1的值。这时计数小于宏定义SIDE_TABLE_DEALLOCATING == 1,就不进行减少计数的操作,直接标记对象正在析构

看到release的代码就会发现在上面代码中宏定义SIDE_TABLE_DEALLOCATING体现出了苹果这个心机婊的用心之深。通常而言,即便引用计数只有8位的占用,在剔除了首位越界标记以及后两位后,其最大取值为2^5-1 == 31位。通常来说,如果不是项目中block不加限制的引用,是很难达到这么多的引用量的。因此占用了SIDE_TABLE_DEALLOCATING位不仅减少了额外占用的标记变量内存,还能以作为引用计数是否归零的判断

weak

最开始的时候没打算讲weak这个修饰,不过因为dealloc方法本身涉及到了弱引用对象置空的操作,以及retain过程中的对象也跟weak有关系的情况下,简单的说说weak的操作

bool objc_object::sidetable_isWeaklyReferenced()
{
    bool result = false;

    SideTable& table = SideTables()[this];
    table.lock();

    RefcountMap::iterator it = table.refcnts.find(this);
    if (it != table.refcnts.end()) {
        result = it->second & SIDE_TABLE_WEAKLY_REFERENCED;
    }

    table.unlock();

    return result;
}

weakstrong共用一套引用计数设计,因此两者的赋值操作都要设置计数表,只是weak修饰的对象的引用计数对象会被设置SIDE_TABLE_WEAKLY_REFERENCED位,并且不参与sidetable_retainCount函数中的计数计算而已

void objc_object::sidetable_setWeaklyReferenced_nolock()
{
#if SUPPORT_NONPOINTER_ISA
    assert(!isa.indexed);
#endif

    SideTable& table = SideTables()[this];

    table.refcnts[this] |= SIDE_TABLE_WEAKLY_REFERENCED;
}

另一个弱引用设置方法,相比上一个方法去掉了自旋锁加锁操作

dealloc

dealloc是重量级的方法之一,不过由于函数内部调用层次过多,这里不多阐述。实现代码在objc-object.h798行,可以自行到官网下载源码后研读

__unsafe_unretained

其实写了这么多,终于把本文的主角给讲出来了。在iOS5的时候,苹果正式推出了ARC机制,伴随的是上面的weakstrong等新修饰符,当然还有一个不常用的__unsafe_unretained

  • weak
    修饰的对象在指向的内存被释放后会被自动置为nil
  • strong
    持有指向的对象,会让引用计数+1
  • __unsafe_unretained
    不引用指向的对象。但在对象内存被释放掉后,依旧指向内存地址,等同于assign,但是只能修饰对象

在机器上保证应用能保持在55帧以上的速率会让应用看起来如丝绸般顺滑,但是稍有不慎,稍微降到50~55之间都有很大的可能展现出卡顿的现象。这里不谈及图像渲染、数据大量处理等耳闻能详的性能恶鬼,说说Model所造成的损耗。

如前面所说的,在ARC环境下,对象的默认修饰为strong,这意味着这么一段代码:

@protocol RegExpCheck

@property (nonatomic, copy) NSString * regExp;

- (BOOL)validRegExp;

@end

- (BOOL)valid: (NSArray<id<RegExpCheck>> *)params {
    for (id<RegExpCheck> item in params) {
        if (![item validRegExp]) { return NO; }
    }
    return YES;
}

把这段代码改为编译期间插入retainrelease方法后的代码如下:

- (BOOL)valid: (NSArray<id<RegExpCheck>> *)params {
    for (id<RegExpCheck> item in params) {
        [item retain];
        if (![item validRegExp]) { 
            [item release];
            return NO;
        }
        [item release];
    }
    return YES;
}

遍历操作在项目中出现的概率绝对排的上前列,那么上面这个方法在调用期间会调用params.countretainrelease函数。通常来说,每一个对象的遍历次数越多,这些函数调用的损耗就越大。如果换做__unsafe_unretained修饰对象,那么这部分的调用损耗就被节省下来,这也是笔者朋友改进的手段

尾话

首先要承认,相比起其他性能恶鬼改进的优化,使用__unsafe_unretained带来的收益几乎微乎其微,因此笔者并不是很推荐用这种高成本低回报的方式优化项目,起码在性能恶鬼大头解决之前不推荐,但是去学习内存管理底层的知识可以帮助我们站在更高的地方看待开发。

ps:在朋友的坚持下,可耻的取消了代码链接

上一篇:消息机制
下一篇:分类为什么不生成setter和getter

转载请注明本文作者和地址

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

推荐阅读更多精彩内容