OC中的load和initialize方法

load

load方法在這個文件被程序裝載時調用(這個類被加入內存的時候調用),這與這個類是否被用到無關,因此load方法總是在main函數之前調用。

如果一個類實現了load方法,在調用這個方法前會首先調用父類的load方法。而且這個過程是自動完成的,并不需要我們手動實現:


// In Parent.m

+ (void)load {

NSLog(@"Load Class Parent");

}

// In Child.m,繼承自Parent

+ (void)load {

NSLog(@"Load Class Child");

}

// In Child+load.m,Child類的分類

+ (void)load {

NSLog(@"Load Class Child+load");

}

// 運行結果:

/*

2016-02-28 10:32:25.284 load[11789:1435378] Load Class Parent

2016-02-28 10:32:25.284 load[11789:1435378] Load Class Child

2016-02-28 10:32:25.284 load[11789:1435378] Load Class Child+load

*/



如果一個類沒有實現load方法,那么就不會調用它父類的load方法,這一點與正常的類繼承和方法調用不一樣,需要額外注意一下。


load方法調用時,系統處于脆弱狀態,如果調用別的類的方法,且該方法依賴于那個類的load方法進行初始化設置,那么必須確保那個類的load方法已經調用了,比如demo中的這段代碼,打印出的字符串就為null:


// In Child.m

+ (void)load {

NSLog(@"Load Class Child");

Other *other = [Other new];

[other originalFunc];

// 如果不先調用other的load,下面這行代碼就無效,打印出null

[Other printName];

}


案例

由于調用load方法時的環境很不安全,我們應該盡量減少load方法的邏輯。另一個原因是load方法是線程安全的,它內部使用了鎖,所以我們應該避免線程阻塞在load方法中。一個常見的使用場景是在load方法中實現Method Swizzle:

// In Other.m

+ (void)load {

Method originalFunc = class_getInstanceMethod([self class], @selector(originalFunc));

Method swizzledFunc = class_getInstanceMethod([self class], @selector(swizzledFunc));

method_exchangeImplementations(originalFunc, swizzledFunc);

}

在Child類的load方法中,由于還沒調用Other的load方法,所以輸出結果是"Original Output",而在main函數中,輸出結果自然就變成了"Swizzled Output"。

一般來說,除了Method Swizzle,別的邏輯都不應該放在load方法中實現。


initialize

與load方法類似的是,在initialize方法內部也會調用父類的方法,而且不需要我們顯示的寫出來。與load方法不同之處在于,即使子類沒有實現initialize方法,也會調用父類的方法,這會導致一個很嚴重的問題:


// In Parent.m

+ (void)initialize {

NSLog(@"Initialize Parent, caller Class %@", [self class]);

}

// In Child.m

// 注釋掉initialize方法

// In main.m

Child *child = [Child new];

運行后發現父類的initialize方法竟然調用了兩次:

2016-02-28 12:11:25.163 load[12772:1509345] Initialize Parent, caller Class Parent

2016-02-28 12:11:25.163 load[12772:1509345] Initialize Parent, caller Class Child


這是因為在創建子類對象時,首先要創建父類對象,所以會調用一次父類的initialize方法,然后創建子類時,盡管自己沒有實現initialize方法,但還是會調用到父類的方法。

雖然initialize方法對一個類而言只會調用一次,但這里由于出現了兩個類,所以調用兩次符合規則,但不符合我們的需求。正確使用initialize方法的姿勢如下:

// In Parent.m

+ (void)initialize {

if (self == [Parent class]) {

NSLog(@"Initialize Parent, caller Class %@", [self class]);

}

}

加上判斷后,就不會因為子類而調用到自己的initialize方法了。

案例

initialize方法主要用來對一些不方便在編譯期初始化的對象進行賦值,比如NSMutableArray這種類型的實例化依賴于runtime的消息發送,所以顯然無法在編譯器初始化:


// In Parent.m

static int someNumber = 0; ? ? // int類型可以在編譯期賦值

static NSMutableArray *someObjects;

+ (void)initialize {

if (self == [Parent class]) {

// 不方便編譯期復制的對象在這里賦值

someObjects = [[NSMutableArray alloc] init];

}

}


總結

load和initialize方法都會在實例化對象之前調用,以main函數為界,前者在main函數之前調用,后者在之后調用。這兩個方法會被自動調用,不能手動調用它們。

load和initialize方法都不用顯示的調用父類的方法而是自動調用,即使子類沒有initialize方法也會調用父類的方法,而load方法則不會調用父類。

load方法通常用來進行Method Swizzle,initialize方法一般用于初始化全局變量或靜態變量。

load和initialize方法內部使用了鎖,因此它們是線程安全的。實現時要盡可能保持簡單,避免阻塞線程,不要再使用鎖。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容

  • OC中有兩個特殊的類方法,分別是load和initialize。本文總結一下這兩個方法的區別于聯系、使用場景和注意...
    RobinYu閱讀 442評論 0 3
  • load,再啟動之前調用 1:父類 2:子類 3:分類 根據函數地址調用 initialize,在初次使用類對象的...
    哈哈哈我的簡書賬號閱讀 375評論 1 1
  • OC中有兩個特殊的類方法,分別是load和initialize。本文總結一下這兩個方法的區別于聯系、使用場景和注意...
    凱旋之歌閱讀 1,805評論 2 2
  • 注:本人是轉載供自己總結用, 做個筆記。 load方法。 在main函數前調用,如果一個類實現了load方法。在調...
    Fs_purple閱讀 342評論 0 4
  • 坐在烏鎮水邊的長椅上,眼前是落日下的余暉,晚霞在河道的那頭,我在河道的這頭,幾株樹影,暮色垂臨,很想就這樣靠著發一...
    影秋千閱讀 497評論 0 5