解读OC中的load和initialize
在 Objective-C 中,NSObject 是絕大多數(shù)類的基類。而在 NSObject 中有兩個(gè)類方法 load 和 initialize,那這兩個(gè)方法是在什么時(shí)機(jī)被調(diào)用呢?父類、Category 的調(diào)用順序又是怎樣的呢?本文解讀一下這兩個(gè)方法的區(qū)別于聯(lián)系及使用場景。
Load
load方法在這個(gè)文件被程序裝載時(shí)調(diào)用。只要是在Compile Sources中出現(xiàn)的文件總是會被裝載,這與這個(gè)類是否被用到無關(guān),因此load方法總是在main函數(shù)之前調(diào)用。這個(gè)很關(guān)鍵,也容易認(rèn)知出錯
調(diào)用順序
如果一個(gè)類實(shí)現(xiàn)了load方法,在調(diào)用這個(gè)方法前會首先調(diào)用父類的load方法。而且這個(gè)過程是自動完成的,并不需要我們手動實(shí)現(xiàn):
父類 -> 子類 -> 父類的Category -> 子類的Category -> Main
// 在 People.m + (void)load {NSLog(@"加載Load方法: People"); }// 在 People+Category.m,People的分類 + (void)load {NSLog(@"加載Load方法: People+Category"); }// 在 Student.m,繼承自Parent + (void)load {NSLog(@"加載Load方法: Student"); }// 在 Student+Category.m,Student的分類 + (void)load {NSLog(@"Load Class Child+load"); }// 運(yùn)行結(jié)果: 2018-12-21 11:27:58.392283+0800 [33801:3250290] 加載Load方法: People 2018-12-21 11:27:58.392825+0800 [33801:3250290] 加載Load方法: Student 2018-12-21 11:27:58.393400+0800 [33801:3250290] 加載Load方法: People+Category 2018-12-21 11:27:58.393520+0800 [33801:3250290] 加載Load方法: Student+Category 2018-12-21 11:27:58.393672+0800 [33801:3250290] Main開始執(zhí)行====>使用場景
由于load方法是線程安全的,它內(nèi)部使用了鎖,所以我們應(yīng)該避免線程阻塞在load方法中。常見的使用場景是在load方法中實(shí)現(xiàn)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方法中,由于還沒調(diào)用Other的load方法,所以輸出結(jié)果是"Original Output",而在main函數(shù)中,輸出結(jié)果自然就變成了"Swizzled Output"。
一般來說,除了Method Swizzle,別的邏輯都不應(yīng)該放在load方法中實(shí)現(xiàn)。
initialize
這個(gè)方法在第一次給某個(gè)類發(fā)送消息時(shí)調(diào)用(比如實(shí)例化一個(gè)對象),并且只會調(diào)用一次。initialize方法實(shí)際上是一種惰性調(diào)用,也就是說如果一個(gè)類一直沒被用到,那它的initialize方法也不會被調(diào)用,這一點(diǎn)有利于節(jié)約資源。
調(diào)用順序
// 在 People.m + (void)initialize {NSLog(@"加載People 的initialize方法: %@", [self class]); }// 在 Student.m + (void)initialize {NSLog(@"加載Student 的initialize方法: %@", [self class]); }// In main.m Student *student = [Student new]; // 運(yùn)行結(jié)果:1: 沒有注釋Student的initialize方法 2018-12-21 11:42:56.694261+0800 [34107:3314744] 加載People 的initialize方法: People 2018-12-21 11:42:56.694433+0800 [34107:3314744] 加載People 的initialize方法: Student2: 注釋Student的initialize方法 2018-12-21 11:42:56.694261+0800 [34107:3314744] 加載People 的initialize方法: People 2018-12-21 11:42:56.694433+0800 [34107:3314744] 加載People 的initialize方法: Student運(yùn)行后發(fā)現(xiàn)父類的initialize方法竟然調(diào)用了兩次:
這是因?yàn)樵趧?chuàng)建子類對象時(shí),首先要創(chuàng)建父類對象,所以會調(diào)用一次父類的initialize方法,然后創(chuàng)建子類時(shí),盡管自己沒有實(shí)現(xiàn)initialize方法,但還是會調(diào)用到父類的方法。
雖然initialize方法對一個(gè)類而言只會調(diào)用一次,但這里由于出現(xiàn)了兩個(gè)類,所以調(diào)用兩次符合規(guī)則,但不符合我們的需求。正確使用initialize方法的姿勢如下:
// In People.m + (void)initialize {if (self == [People class]) {//TODO...} }使用場景
initialize方法主要用來對一些不方便在編譯期初始化的對象進(jìn)行賦值。比如NSMutableArray這種類型的實(shí)例化依賴于runtime的消息發(fā)送,所以顯然無法在編譯器初始化:
總結(jié)
轉(zhuǎn)載于:https://www.cnblogs.com/fengtengfei/p/10154964.html
總結(jié)
以上是生活随笔為你收集整理的解读OC中的load和initialize的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Linux中几个实用快捷键
- 下一篇: 模拟器登陆