NSObject的load和initialize方法

在Objective-C中,NSObject是根類(lèi)蜈缤,而NSObject.h的頭文件中前兩個(gè)方法就是load和initialize兩個(gè)類(lèi)方法,本篇文章就對(duì)這兩個(gè)方法做下說(shuō)明和整理冯挎。

0. 概述
Objective-C作為一門(mén)面向?qū)ο笳Z(yǔ)言底哥,有類(lèi)和對(duì)象的概念。編譯后房官,類(lèi)相關(guān)的數(shù)據(jù)結(jié)構(gòu)會(huì)保留在目標(biāo)文件中趾徽,在運(yùn)行時(shí)得到解析和使用。在應(yīng)用程序運(yùn)行起來(lái)的時(shí)候翰守,類(lèi)的信息會(huì)有加載和初始化過(guò)程孵奶。

其實(shí)在Java語(yǔ)言中也有類(lèi)似的過(guò)程,JVM的ClassLoader也對(duì)類(lèi)進(jìn)行了加載潦俺、連接拒课、初始化徐勃。

就像Application有生命周期回調(diào)方法一樣事示,在Objective-C的類(lèi)被加載和初始化的時(shí)候,也可以收到方法回調(diào)僻肖,可以在適當(dāng)?shù)那闆r下做一些定制處理肖爵。而這正是load和initialize方法可以幫我們做到的。

+ (void)load;
+ (void)initialize;

可以看到這兩個(gè)方法都是以“+”開(kāi)頭的類(lèi)方法臀脏,返回為空劝堪。通常情況下,我們?cè)陂_(kāi)發(fā)過(guò)程中可能不必關(guān)注這兩個(gè)方法揉稚。如果有需要定制秒啦,我們可以在自定義的NSObject子類(lèi)中給出這兩個(gè)方法的實(shí)現(xiàn),這樣在類(lèi)的加載和初始化過(guò)程中搀玖,自定義的方法可以得到調(diào)用余境。
從如上聲明上來(lái)看,也許這兩個(gè)方法和其它的類(lèi)方法相比沒(méi)什么特別灌诅。但是芳来,這兩個(gè)方法具有一定的“特殊性”,這也是這兩個(gè)方法經(jīng)常會(huì)被放在一起特殊提到的原因猜拾。詳細(xì)請(qǐng)看如下幾小節(jié)的整理即舌。

1. load和initialize的共同特點(diǎn)
load和initialize有很多共同特點(diǎn),下面簡(jiǎn)單列一下:

  • 在不考慮開(kāi)發(fā)者主動(dòng)使用的情況下挎袜,系統(tǒng)最多會(huì)調(diào)用一次
  • 如果父類(lèi)和子類(lèi)都被調(diào)用顽聂,父類(lèi)的調(diào)用一定在子類(lèi)之前
  • 都是為了應(yīng)用運(yùn)行提前創(chuàng)建合適的運(yùn)行環(huán)境
  • 在使用時(shí)都不要過(guò)重地依賴于這兩個(gè)方法肥惭,除非真正必要

2. load方法相關(guān)要點(diǎn)
廢話不多說(shuō),直接上要點(diǎn)列表:

  • 調(diào)用時(shí)機(jī)比較早紊搪,運(yùn)行環(huán)境有不確定因素务豺。具體說(shuō)來(lái),在iOS上通常就是App啟動(dòng)時(shí)進(jìn)行加載嗦明,但當(dāng)load調(diào)用的時(shí)候笼沥,并不能保證所有類(lèi)都加載完成且可用,必要時(shí)還要自己負(fù)責(zé)做auto release處理娶牌。
  • 補(bǔ)充上面一點(diǎn)奔浅,對(duì)于有依賴關(guān)系的兩個(gè)庫(kù)中,被依賴的類(lèi)的load會(huì)優(yōu)先調(diào)用诗良。但在一個(gè)庫(kù)之內(nèi)汹桦,調(diào)用順序是不確定的。
  • 對(duì)于一個(gè)類(lèi)而言鉴裹,沒(méi)有l(wèi)oad方法實(shí)現(xiàn)就不會(huì)調(diào)用舞骆,不會(huì)考慮對(duì)NSObject的繼承径荔。
  • 一個(gè)類(lèi)的load方法不用寫(xiě)明[super load]督禽,父類(lèi)就會(huì)收到調(diào)用,并且在子類(lèi)之前狈惫。
  • Category的load也會(huì)收到調(diào)用,但順序上在主類(lèi)的load調(diào)用之后鹦马。
  • 不會(huì)直接觸發(fā)initialize的調(diào)用胧谈。

3. initialize方法相關(guān)要點(diǎn)
同樣,直接整理要點(diǎn):

  • initialize的自然調(diào)用是在第一次主動(dòng)使用當(dāng)前類(lèi)的時(shí)候(lazy荸频,這一點(diǎn)和Java類(lèi)的“clinit”的很像)菱肖。
  • 在initialize方法收到調(diào)用時(shí),運(yùn)行環(huán)境基本健全旭从。
  • initialize的運(yùn)行過(guò)程中是能保證線程安全的稳强。
  • 和load不同,即使子類(lèi)不實(shí)現(xiàn)initialize方法遇绞,會(huì)把父類(lèi)的實(shí)現(xiàn)繼承過(guò)來(lái)調(diào)用一遍键袱。注意的是在此之前,父類(lèi)的方法已經(jīng)被執(zhí)行過(guò)一次了摹闽,同樣不需要super調(diào)用蹄咖。

由于initialize的這些特點(diǎn),使得其應(yīng)用比load要略微廣泛一些付鹿±教溃可用來(lái)做一些初始化工作蚜迅,或者單例模式的一種實(shí)現(xiàn)方案

4. 原理
“源碼面前沒(méi)有秘密”。最后俊抵,我們來(lái)看看蘋(píng)果開(kāi)放出來(lái)的部分源碼谁不。從中我們也許能明白為什么load和initialize及調(diào)用會(huì)有如上的一些特點(diǎn)。
其中l(wèi)oad是在objc庫(kù)中一個(gè)load_images函數(shù)中調(diào)用的徽诲,先把二進(jìn)制映像文件中的頭信息取出刹帕,再解析和讀出各個(gè)模塊中的類(lèi)定義信息,把實(shí)現(xiàn)了load方法的類(lèi)和Category記錄下來(lái)谎替,最后統(tǒng)一執(zhí)行調(diào)用偷溺。
其中的prepare_load_methods函數(shù)實(shí)現(xiàn)如下:

void prepare_load_methods(header_info *hi)
{
    Module mods;
    unsigned int midx;
    if (_objcHeaderIsReplacement(hi)) {
        return;
    }
  
    mods = hi->mod_ptr;
    for (midx = 0; midx < hi->mod_count; midx += 1)
    {
        unsigned int index;
  
        if (mods[midx].symtab == nil)
            continue;
  
        for (index = 0; index < mods[midx].symtab->cls_def_cnt; index += 1)
        {
            Class cls = (Class)mods[midx].symtab->defs[index];
            if (cls->info & CLS_CONNECTED) {
                schedule_class_load(cls);
            }
        }
    }
    mods = hi->mod_ptr;
  
    midx = (unsigned int)hi->mod_count;
    while (midx-- > 0) {
        unsigned int index;
        unsigned int total;
        Symtab symtab = mods[midx].symtab;
  
        if (mods[midx].symtab == nil)
            continue;
        total = mods[midx].symtab->cls_def_cnt +
            mods[midx].symtab->cat_def_cnt;
  
        index = total;
        while (index-- > mods[midx].symtab->cls_def_cnt) {
            old_category *cat = (old_category *)symtab->defs[index];
            add_category_to_loadable_list((Category)cat);
        }
    }
}

這大概就是主類(lèi)中的load方法先于category的原因。再看下面這段:

static void schedule_class_load(Class cls)
{
    if (cls->info & CLS_LOADED) return;
    if (cls->superclass) schedule_class_load(cls->superclass);
    add_class_to_loadable_list(cls);
    cls->info |= CLS_LOADED;
}

這正是父類(lèi)load方法優(yōu)先于子類(lèi)調(diào)用的原因钱贯。

再來(lái)看下initialize調(diào)用相關(guān)的源碼挫掏。objc的庫(kù)里有一個(gè)_class_initialize方法實(shí)現(xiàn),如下:

void _class_initialize(Class cls)
{
    assert(!cls->isMetaClass());
  
    Class supercls;
    BOOL reallyInitialize = NO;
  
    supercls = cls->superclass;
    if (supercls  &&  !supercls->isInitialized()) {
        _class_initialize(supercls);
    }
  
    monitor_enter(&classInitLock);
    if (!cls->isInitialized() && !cls->isInitializing()) {
        cls->setInitializing();
        reallyInitialize = YES;
    }
    monitor_exit(&classInitLock);
  
    if (reallyInitialize) {
        _setThisThreadIsInitializingClass(cls);
  
        if (PrintInitializing) {
            _objc_inform("INITIALIZE: calling +[%s initialize]",
                         cls->nameForLogging());
        }
  
        ((void(*)(Class, SEL))objc_msgSend)(cls, SEL_initialize);
  
        if (PrintInitializing) {
            _objc_inform("INITIALIZE: finished +[%s initialize]",
                         cls->nameForLogging());
        }
  
        monitor_enter(&classInitLock);
        if (!supercls  ||  supercls->isInitialized()) {
            _finishInitializing(cls, supercls);
        } else {
            _finishInitializingAfter(cls, supercls);
        }
        monitor_exit(&classInitLock);
        return;
    }
  
    else if (cls->isInitializing()) {
        if (_thisThreadIsInitializingClass(cls)) {
            return;
        } else {
            monitor_enter(&classInitLock);
            while (!cls->isInitialized()) {
                monitor_wait(&classInitLock);
            }
            monitor_exit(&classInitLock);
            return;
        }
    }
  
    else if (cls->isInitialized()) {
        return;
    }
  
    else {
        _objc_fatal("thread-safe class init in objc runtime is buggy!");
    }
}

在這段代碼里秩命,我們能看到initialize的調(diào)用順序和線程安全性尉共。

轉(zhuǎn)自:cocoachina,原作者:三石

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末弃锐,一起剝皮案震驚了整個(gè)濱河市袄友,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌拿愧,老刑警劉巖杠河,帶你破解...
    沈念sama閱讀 207,113評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異浇辜,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)唾戚,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,644評(píng)論 2 381
  • 文/潘曉璐 我一進(jìn)店門(mén)柳洋,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人叹坦,你說(shuō)我怎么就攤上這事熊镣。” “怎么了募书?”我有些...
    開(kāi)封第一講書(shū)人閱讀 153,340評(píng)論 0 344
  • 文/不壞的土叔 我叫張陵绪囱,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我莹捡,道長(zhǎng)鬼吵,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 55,449評(píng)論 1 279
  • 正文 為了忘掉前任篮赢,我火速辦了婚禮齿椅,結(jié)果婚禮上琉挖,老公的妹妹穿的比我還像新娘。我一直安慰自己涣脚,他們只是感情好示辈,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,445評(píng)論 5 374
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著遣蚀,像睡著了一般矾麻。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上芭梯,一...
    開(kāi)封第一講書(shū)人閱讀 49,166評(píng)論 1 284
  • 那天射富,我揣著相機(jī)與錄音,去河邊找鬼粥帚。 笑死胰耗,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的芒涡。 我是一名探鬼主播柴灯,決...
    沈念sama閱讀 38,442評(píng)論 3 401
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼费尽!你這毒婦竟也來(lái)了赠群?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,105評(píng)論 0 261
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤旱幼,失蹤者是張志新(化名)和其女友劉穎查描,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體柏卤,經(jīng)...
    沈念sama閱讀 43,601評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡冬三,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,066評(píng)論 2 325
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了缘缚。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片勾笆。...
    茶點(diǎn)故事閱讀 38,161評(píng)論 1 334
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖桥滨,靈堂內(nèi)的尸體忽然破棺而出窝爪,到底是詐尸還是另有隱情,我是刑警寧澤齐媒,帶...
    沈念sama閱讀 33,792評(píng)論 4 323
  • 正文 年R本政府宣布蒲每,位于F島的核電站,受9級(jí)特大地震影響喻括,放射性物質(zhì)發(fā)生泄漏邀杏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,351評(píng)論 3 307
  • 文/蒙蒙 一双妨、第九天 我趴在偏房一處隱蔽的房頂上張望淮阐。 院中可真熱鬧叮阅,春花似錦、人聲如沸泣特。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,352評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)状您。三九已至勒叠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間膏孟,已是汗流浹背眯分。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 31,584評(píng)論 1 261
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留柒桑,地道東北人弊决。 一個(gè)月前我還...
    沈念sama閱讀 45,618評(píng)論 2 355
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像魁淳,于是被迫代替她去往敵國(guó)和親飘诗。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,916評(píng)論 2 344

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

  • 在Objective-C中界逛,NSObject是根類(lèi)昆稿,而NSObject.h的頭文件中前兩個(gè)方法就是load和ini...
    SuAdrenine閱讀 319評(píng)論 0 1
  • 0. 概述Objective-C作為一門(mén)面向?qū)ο笳Z(yǔ)言,有類(lèi)和對(duì)象的概念息拜。編譯后溉潭,類(lèi)相關(guān)的數(shù)據(jù)結(jié)構(gòu)會(huì)保留在目標(biāo)文件中...
    Crazy2015閱讀 375評(píng)論 0 0
  • 在Objective-C中,NSObject是根類(lèi)少欺,而NSObject.h的頭文件中前兩個(gè)方法就是load和ini...
    曲年閱讀 283評(píng)論 0 1
  • 在Objective-C中喳瓣,NSObject是根類(lèi),而NSObject.h的頭文件中前兩個(gè)方法就是load和ini...
    ImmortalSummer閱讀 406評(píng)論 0 0
  • 在Objective-C中狈茉,NSObject是根類(lèi)夫椭,而NSObject.h的頭文件中前兩個(gè)方法就是load和ini...
    ch32053閱讀 804評(píng)論 0 9