load和initialize的區別

在Objective-C中,NSObject是根類,而NSObject.h的頭文件中前兩個方法就是load和initialize兩個類方法,本篇文章就對這兩個方法做下說明和整理。

0. 概述

Objective-C作為一門面向對象語言,有類和對象的概念。編譯后,類相關的數據結構會保留在目標文件中,在運行時得到解析和使用。在應用程序運行起來的時候,類的信息會有加載和初始化過程。

其實在Java語言中也有類似的過程,JVM的ClassLoader也對類進行了加載、連接、初始化。

就像Application有生命周期回調方法一樣,在Objective-C的類被加載和初始化的時候,也可以收到方法回調,可以在適當的情況下做一些定制處理。而這正是load和initialize方法可以幫我們做到的。

+?(void)load;

+?(void)initialize;

可以看到這兩個方法都是以“+”開頭的類方法,返回為空。通常情況下,我們在開發過程中可能不必關注這兩個方法。如果有需要定制,我們可以在自定義的NSObject子類中給出這兩個方法的實現,這樣在類的加載和初始化過程中,自定義的方法可以得到調用。

從如上聲明上來看,也許這兩個方法和其它的類方法相比沒什么特別。但是,這兩個方法具有一定的“特殊性”,這也是這兩個方法經常會被放在一起特殊提到的原因。詳細請看如下幾小節的整理。

1. load和initialize的共同特點

load和initialize有很多共同特點,下面簡單列一下:

在不考慮開發者主動使用的情況下,系統最多會調用一次

如果父類和子類都被調用,父類的調用一定在子類之前

都是為了應用運行提前創建合適的運行環境

在使用時都不要過重地依賴于這兩個方法,除非真正必要

2. load方法相關要點

廢話不多說,直接上要點列表:

調用時機比較早,運行環境有不確定因素。具體說來,在iOS上通常就是App啟動時進行加載,但當load調用的時候,并不能保證所有類都加載完成且可用,必要時還要自己負責做auto release處理。

補充上面一點,對于有依賴關系的兩個庫中,被依賴的類的load會優先調用。但在一個庫之內,調用順序是不確定的。

對于一個類而言,沒有load方法實現就不會調用,不會考慮對NSObject的繼承。

一個類的load方法不用寫明[super load],父類就會收到調用,并且在子類之前。

Category的load也會收到調用,但順序上在主類的load調用之后。

不會直接觸發initialize的調用。

3. initialize方法相關要點

同樣,直接整理要點:

initialize的自然調用是在第一次主動使用當前類的時候(lazy,這一點和Java類的“clinit”的很像)。

在initialize方法收到調用時,運行環境基本健全。

initialize的運行過程中是能保證線程安全的。

和load不同,即使子類不實現initialize方法,會把父類的實現繼承過來調用一遍。注意的是在此之前,父類的方法已經被執行過一次了,同樣不需要super調用。

由于initialize的這些特點,使得其應用比load要略微廣泛一些??捎脕碜鲆恍┏跏蓟ぷ?,或者單例模式的一種實現方案。

4. 原理

“源碼面前沒有秘密”。最后,我們來看看蘋果開放出來的部分源碼。從中我們也許能明白為什么load和initialize及調用會有如上的一些特點。

其中load是在objc庫中一個load_images函數中調用的,先把二進制映像文件中的頭信息取出,再解析和讀出各個模塊中的類定義信息,把實現了load方法的類和Category記錄下來,最后統一執行調用。

其中的prepare_load_methods函數實現如下:

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);

}

}

}

這大概就是主類中的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;

}

這正是父類load方法優先于子類調用的原因。

再來看下initialize調用相關的源碼。objc的庫里有一個_class_initialize方法實現,如下:

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!");

}

}

轉載自三石道博客

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,247評論 6 543
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,520評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,362評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,805評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,541評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,896評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,887評論 3 447
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,062評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,608評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,356評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,555評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,077評論 5 364
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,769評論 3 349
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,175評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,489評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,289評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,516評論 2 379

推薦閱讀更多精彩內容