NSObject的load和initialize方法

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

  1. 概述

Objective-C作為一門面向?qū)ο笳Z言芭梯,有類和對象的概念茅坛。編譯后音半,類相關的數(shù)據(jù)結構會保留在目標文件中,在運行時得到解析和使用灰蛙。在應用程序運行起來的時候祟剔,類的信息會有加載和初始化過程。

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

就像Application有生命周期回調(diào)方法一樣叛薯,在Objective-C的類被加載和初始化的時候,也可以收到方法回調(diào)笙纤,可以在適當?shù)那闆r下做一些定制處理耗溜。而這正是load和initialize方法可以幫我們做到的。
<pre><code>+ (void)load;

  • (void)initialize;</pre></code>
    可以看到這兩個方法都是以“+”開頭的類方法省容,返回為空抖拴。通常情況下,我們在開發(fā)過程中可能不必關注這兩個方法。如果有需要定制阿宅,我們可以在自定義的NSObject子類中給出這兩個方法的實現(xiàn)候衍,這樣在類的加載和初始化過程中,自定義的方法可以得到調(diào)用洒放。

從如上聲明上來看蛉鹿,也許這兩個方法和其它的類方法相比沒什么特別。但是往湿,這兩個方法具有一定的“特殊性”妖异,這也是這兩個方法經(jīng)常會被放在一起特殊提到的原因。詳細請看如下幾小節(jié)的整理领追。

  1. load和initialize的共同特點

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

在不考慮開發(fā)者主動使用的情況下,系統(tǒng)最多會調(diào)用一次
如果父類和子類都被調(diào)用绒窑,父類的調(diào)用一定在子類之前
都是為了應用運行提前創(chuàng)建合適的運行環(huán)境
在使用時都不要過重地依賴于這兩個方法矩乐,除非真正必要

  1. load方法相關要點

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

調(diào)用時機比較早回论,運行環(huán)境有不確定因素。具體說來分歇,在iOS上通常就是App啟動時進行加載傀蓉,但當load調(diào)用的時候,并不能保證所有類都加載完成且可用职抡,必要時還要自己負責做auto release處理葬燎。
補充上面一點,對于有依賴關系的兩個庫中缚甩,被依賴的類的load會優(yōu)先調(diào)用谱净。但在一個庫之內(nèi),調(diào)用順序是不確定的擅威。
對于一個類而言壕探,沒有l(wèi)oad方法實現(xiàn)就不會調(diào)用,不會考慮對NSObject的繼承郊丛。
一個類的load方法不用寫明[super load]李请,父類就會收到調(diào)用,并且在子類之前厉熟。
Category的load也會收到調(diào)用导盅,但順序上在主類的load調(diào)用之后。
不會直接觸發(fā)initialize的調(diào)用揍瑟。

  1. initialize方法相關要點

同樣白翻,直接整理要點:

initialize的自然調(diào)用是在第一次主動使用當前類的時候(lazy,這一點和Java類的“clinit”的很像)绢片。
在initialize方法收到調(diào)用時滤馍,運行環(huán)境基本健全岛琼。
initialize的運行過程中是能保證線程安全的。
和load不同纪蜒,即使子類不實現(xiàn)initialize方法衷恭,會把父類的實現(xiàn)繼承過來調(diào)用一遍。注意的是在此之前纯续,父類的方法已經(jīng)被執(zhí)行過一次了随珠,同樣不需要super調(diào)用。
由于initialize的這些特點猬错,使得其應用比load要略微廣泛一些窗看。可用來做一些初始化工作倦炒,或者單例模式的一種實現(xiàn)方案显沈。

  1. 原理

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

其中l(wèi)oad是在objc庫中一個load_images函數(shù)中調(diào)用的鳖藕,先把二進制映像文件中的頭信息取出魔慷,再解析和讀出各個模塊中的類定義信息,把實現(xiàn)了load方法的類和Category記錄下來著恩,最后統(tǒng)一執(zhí)行調(diào)用院尔。

其中的prepare_load_methods函數(shù)實現(xiàn)如下:
<pre><code>
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);
}
}
}
</pre></code>
這大概就是主類中的load方法先于category的原因。再看下面這段:
<pre><code>
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;
}
</pre></code>
這正是父類load方法優(yōu)先于子類調(diào)用的原因喉誊。

再來看下initialize調(diào)用相關的源碼邀摆。objc的庫里有一個_class_initialize方法實現(xiàn),如下:
<pre><code>
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!");
}
}
</pre></code>
在這段代碼里伍茄,我們能看到initialize的調(diào)用順序和線程安全性栋盹。
(本文作者:三石)
轉(zhuǎn)載自:http://www.molotang.com/articles/1929.html

最后編輯于
?著作權歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市幻林,隨后出現(xiàn)的幾起案子贞盯,更是在濱河造成了極大的恐慌,老刑警劉巖沪饺,帶你破解...
    沈念sama閱讀 221,198評論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件躏敢,死亡現(xiàn)場離奇詭異,居然都是意外死亡整葡,警方通過查閱死者的電腦和手機件余,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評論 3 398
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人啼器,你說我怎么就攤上這事旬渠∩蔡剩” “怎么了姆涩?”我有些...
    開封第一講書人閱讀 167,643評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長垒玲。 經(jīng)常有香客問我损谦,道長岖免,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,495評論 1 296
  • 正文 為了忘掉前任照捡,我火速辦了婚禮颅湘,結果婚禮上,老公的妹妹穿的比我還像新娘栗精。我一直安慰自己闯参,他們只是感情好,可當我...
    茶點故事閱讀 68,502評論 6 397
  • 文/花漫 我一把揭開白布悲立。 她就那樣靜靜地躺著鹿寨,像睡著了一般。 火紅的嫁衣襯著肌膚如雪薪夕。 梳的紋絲不亂的頭發(fā)上释移,一...
    開封第一講書人閱讀 52,156評論 1 308
  • 那天,我揣著相機與錄音寥殖,去河邊找鬼。 笑死涩蜘,一個胖子當著我的面吹牛嚼贡,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播同诫,決...
    沈念sama閱讀 40,743評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼粤策,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了误窖?” 一聲冷哼從身側響起叮盘,我...
    開封第一講書人閱讀 39,659評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎霹俺,沒想到半個月后柔吼,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 46,200評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡丙唧,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,282評論 3 340
  • 正文 我和宋清朗相戀三年愈魏,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,424評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡培漏,死狀恐怖溪厘,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情牌柄,我是刑警寧澤畸悬,帶...
    沈念sama閱讀 36,107評論 5 349
  • 正文 年R本政府宣布,位于F島的核電站珊佣,受9級特大地震影響蹋宦,放射性物質(zhì)發(fā)生泄漏彩扔。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,789評論 3 333
  • 文/蒙蒙 一敦捧、第九天 我趴在偏房一處隱蔽的房頂上張望兢卵。 院中可真熱鬧秽荤,春花似錦窃款、人聲如沸晨继。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽驻谆。三九已至胜臊,卻和暖如春象对,著一層夾襖步出監(jiān)牢的瞬間勒魔,已是汗流浹背抚吠。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留萧朝,地道東北人。 一個月前我還...
    沈念sama閱讀 48,798評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像头朱,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,435評論 2 359

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