NSObject +alloc做了什么

窺探

通過(guò)下符號(hào)斷點(diǎn)的方式來(lái)跟蹤 [NSObject alloc]

//下了兩個(gè)符號(hào)斷點(diǎn)
1翩隧、objc_alloc
2友驮、+[NSObject alloc]

//結(jié)果是 1 2

經(jīng)過(guò)編譯器一頓操作,似乎結(jié)合和預(yù)想不是完全吻合

在使用+[NSObject alloc]的時(shí)候?qū)嶋H上調(diào)用的是libonjc.A.dylib objc_alloc异旧,并不是預(yù)想的libonjc.A.dylib objc_msgSend,這里看一llvm的源碼,對(duì)alloc做了處理(第一次調(diào)用objc_alloc,第二次調(diào)用+alloc

梳理

1. 從objc_alloc開(kāi)始

// Calls [cls alloc].
id
objc_alloc(Class cls)
{
    return callAlloc(cls, true/*checkNil*/, false/*allocWithZone*/);
}
  • 這里就做一件事情設(shè)置兩個(gè)參數(shù)checkNil = trueallocWithZone = false
  • 初次調(diào)用callAlloc 需要檢查nil,而且不使用zone

2. callAlloc

static id callAlloc(Class cls, bool checkNil, bool allocWithZone=false)
{
    if (slowpath(checkNil && !cls)) return nil;

#if __OBJC2__
    //是否實(shí)現(xiàn) allocWithZone
    if (fastpath(!cls->ISA()->hasCustomAWZ())) {
        //未實(shí)現(xiàn)allocWithZone

        // canAllocFast直接返回了false右遭,直接看else
        if (fastpath(cls->canAllocFast())) {
            // No ctors, raw isa, etc. Go straight to the metal.
            bool dtor = cls->hasCxxDtor();
            id obj = (id)calloc(1, cls->bits.fastInstanceSize());
            if (slowpath(!obj)) return callBadAllocHandler(cls);
            obj->initInstanceIsa(cls, dtor);
            return obj;
        }
        else {
            // 核心 創(chuàng)建實(shí)例
            id obj = class_createInstance(cls, 0);
            if (slowpath(!obj)) return callBadAllocHandler(cls);
            return obj;
        }
    }
#endif

    // No shortcuts available.
    if (allocWithZone) return [cls allocWithZone:nil];
    //+alloc 終于輪到你了
    return [cls alloc];
}

  • hasCustomAWZ() 是檢查是否實(shí)現(xiàn)了 +allocWithZone
  • 沒(méi)有實(shí)現(xiàn)+allocWithZone 就通過(guò) class_createInstance(cls, 0)來(lái)創(chuàng)建實(shí)例
  • 實(shí)現(xiàn)了 +allocWithZone,但還需要allocWithZone == true才會(huì)調(diào)用,首次調(diào)用一定不會(huì)觸發(fā)在objc_alloc中傳了false
  • return [cls alloc];最后這段代碼才是主流程

3. class_createInstance

class_createInstance(Class cls, size_t extraBytes)
{
    //調(diào)用了 私有方法 _class_createInstanceFromZone
    return _class_createInstanceFromZone(cls, extraBytes, nil);
}

id
_class_createInstanceFromZone(Class cls, size_t extraBytes, void *zone, 
                              bool cxxConstruct = true, 
                              size_t *outAllocatedSize = nil)
{
    if (!cls) return nil;

    assert(cls->isRealized());

    // Read class's info bits all at once for performance
    bool hasCxxCtor = cls->hasCxxCtor();  //有c++構(gòu)造函數(shù)
    bool hasCxxDtor = cls->hasCxxDtor(); //有c++析構(gòu)函數(shù)
    bool fast = cls->canAllocNonpointer(); //快速創(chuàng)建窘哈,是否允許非純指針吹榴,也就是isa的優(yōu)化

    //計(jì)算實(shí)例大小
    size_t size = cls->instanceSize(extraBytes);
    if (outAllocatedSize) *outAllocatedSize = size;

    id obj;
    if (!zone  &&  fast) {
        //不使用zone并且支持isa優(yōu)化的對(duì)象分配空間
        obj = (id)calloc(1, size);
        if (!obj) return nil;
        //對(duì)象實(shí)例初始化isa
        obj->initInstanceIsa(cls, hasCxxDtor);
    } 
    else {
        if (zone) {
            //使用zone 分配空間
            obj = (id)malloc_zone_calloc ((malloc_zone_t *)zone, 1, size);
        } else {
            //不使用 zone分配空間
            obj = (id)calloc(1, size);
        }
        if (!obj) return nil;

        //純指針的初始化
        obj->initIsa(cls);
    }

    if (cxxConstruct && hasCxxCtor) {
        obj = _objc_constructOrFree(obj, cls);
    }

    return obj;
}

3. instanceSize

    //拿到類(lèi)對(duì)象的 ro 對(duì)象的原始大小
    uint32_t unalignedInstanceSize() {
        assert(isRealized());
        return data()->ro->instanceSize;
    }

    /*
      字節(jié)對(duì)齊: 為了讓CPU 尋址更簡(jiǎn)單快速,用空間換時(shí)間
       */
    uint32_t alignedInstanceSize() {
        return word_align(unalignedInstanceSize());
    }
    
    //計(jì)算對(duì)象大小
    size_t instanceSize(size_t extraBytes) {
        size_t size = alignedInstanceSize() + extraBytes;
        // CF requires all objects be at least 16 bytes.
        //最小 16個(gè)字節(jié)
        if (size < 16) size = 16;
        return size;
    }

關(guān)于字節(jié)對(duì)齊的內(nèi)容

4. InitIsa

inline void 
objc_object::initIsa(Class cls)
{
    //純指針的初始化 isa 方法
    initIsa(cls, false, false);
}

inline void 
objc_object::initInstanceIsa(Class cls, bool hasCxxDtor)
{
    //非純指針的初始化 isa 方法
    initIsa(cls, true, hasCxxDtor);
}

/*
初始化Isa
參數(shù):
cls 類(lèi)對(duì)象
nonpointer 是否是非指針滚婉,就是經(jīng)過(guò)優(yōu)化的isa
hasCxxDtor 是否存在c++析構(gòu)
*/ 
inline void 
objc_object::initIsa(Class cls, bool nonpointer, bool hasCxxDtor) 
{ 
    assert(!isTaggedPointer()); 
    //
    if (!nonpointer) {
        isa.cls = cls; 
    } else {
        isa_t newisa(0);

        newisa.bits = ISA_MAGIC_VALUE;
        // isa.magic is part of ISA_MAGIC_VALUE
        // isa.nonpointer is part of ISA_MAGIC_VALUE
        newisa.has_cxx_dtor = hasCxxDtor;
        newisa.shiftcls = (uintptr_t)cls >> 3;
        isa = newisa;
    }
}

isa的相關(guān)內(nèi)容

5. +alloc

+ (id)alloc {
    return _objc_rootAlloc(self);
}
id
_objc_rootAlloc(Class cls)
{
    return callAlloc(cls, false/*checkNil*/, true/*allocWithZone*/);
}
  • 未實(shí)現(xiàn)+allocWithZone將會(huì)調(diào)用+ alloc 图筹,然后再次調(diào)用callAlloc,這次參數(shù)不一樣了不需要檢查是否為空让腹,使用zone

總結(jié)

  1. 控制調(diào)用 callAlloc參數(shù)來(lái)控制執(zhí)行路徑远剩,從而實(shí)現(xiàn)不同情況下創(chuàng)建對(duì)象
  2. 對(duì)純指針和非純指針的初始化
  3. 內(nèi)存直接對(duì)齊

附上alloc分析圖


alloc.jpg
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市骇窍,隨后出現(xiàn)的幾起案子瓜晤,更是在濱河造成了極大的恐慌,老刑警劉巖像鸡,帶你破解...
    沈念sama閱讀 212,657評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件活鹰,死亡現(xiàn)場(chǎng)離奇詭異,居然都是意外死亡只估,警方通過(guò)查閱死者的電腦和手機(jī)志群,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,662評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén),熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)蛔钙,“玉大人锌云,你說(shuō)我怎么就攤上這事∮跬眩” “怎么了桑涎?”我有些...
    開(kāi)封第一講書(shū)人閱讀 158,143評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵,是天一觀(guān)的道長(zhǎng)兼贡。 經(jīng)常有香客問(wèn)我攻冷,道長(zhǎng),這世上最難降的妖魔是什么遍希? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,732評(píng)論 1 284
  • 正文 為了忘掉前任等曼,我火速辦了婚禮,結(jié)果婚禮上凿蒜,老公的妹妹穿的比我還像新娘禁谦。我一直安慰自己,他們只是感情好废封,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,837評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布州泊。 她就那樣靜靜地躺著,像睡著了一般漂洋。 火紅的嫁衣襯著肌膚如雪遥皂。 梳的紋絲不亂的頭發(fā)上力喷,一...
    開(kāi)封第一講書(shū)人閱讀 50,036評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音渴肉,去河邊找鬼冗懦。 笑死,一個(gè)胖子當(dāng)著我的面吹牛仇祭,可吹牛的內(nèi)容都是我干的披蕉。 我是一名探鬼主播,決...
    沈念sama閱讀 39,126評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼乌奇,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼没讲!你這毒婦竟也來(lái)了?” 一聲冷哼從身側(cè)響起礁苗,我...
    開(kāi)封第一講書(shū)人閱讀 37,868評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤爬凑,失蹤者是張志新(化名)和其女友劉穎,沒(méi)想到半個(gè)月后试伙,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體嘁信,經(jīng)...
    沈念sama閱讀 44,315評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,641評(píng)論 2 327
  • 正文 我和宋清朗相戀三年疏叨,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了潘靖。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 38,773評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡蚤蔓,死狀恐怖卦溢,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情秀又,我是刑警寧澤单寂,帶...
    沈念sama閱讀 34,470評(píng)論 4 333
  • 正文 年R本政府宣布,位于F島的核電站吐辙,受9級(jí)特大地震影響宣决,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜昏苏,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,126評(píng)論 3 317
  • 文/蒙蒙 一疲扎、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧捷雕,春花似錦、人聲如沸壹甥。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,859評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)句柠。三九已至浦译,卻和暖如春棒假,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背精盅。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,095評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工帽哑, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人叹俏。 一個(gè)月前我還...
    沈念sama閱讀 46,584評(píng)論 2 362
  • 正文 我出身青樓妻枕,卻偏偏與公主長(zhǎng)得像,于是被迫代替她去往敵國(guó)和親粘驰。 傳聞我的和親對(duì)象是個(gè)殘疾皇子屡谐,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,676評(píng)論 2 351

推薦閱讀更多精彩內(nèi)容