iOS Cateogry的深入理解&&load方法調(diào)用&&分類重寫方法的調(diào)用順序(一)

首先先看幾個面試問題

  • Cateogry里面有l(wèi)oad方法么? load方法什么時候調(diào)用?load方法有繼承么拂檩?

1. 新建一個項目嘲碧,并添加TCPerson類,并給TCPerson添加兩個分類

類結(jié)構(gòu)

2.新建一個TCStudent類繼承自TCPerson钉迷,并且給TCStudent也添加兩個分類

image.png

Cateogry里面有l(wèi)oad方法么糠聪?

  • 答:分類里面肯定有l(wèi)oad

#import "TCPerson.h"

@implementation TCPerson
+ (void)load{
    
}
@end
#import "TCPerson+TCtest1.h"

@implementation TCPerson (TCtest1)
+ (void)load{
    
}
@end
#import "TCPerson+TCTest2.h"

@implementation TCPerson (TCTest2)
+ (void)load{
    
}
@end

load方法什么時候調(diào)用舰蟆?

  • load方法在runtime加載類和分類的時候調(diào)用load

#import <Foundation/Foundation.h>

int main(int argc, const char * argv[]) {
    @autoreleasepool {
        
    }
    return 0;
}

@implementation TCPerson
+ (void)load{
    NSLog(@"TCPerson +load");
}
@end


@implementation TCPerson (TCtest1)
+ (void)load{
    NSLog(@"TCPerson (TCtest1) +load");
}
@end
@implementation TCPerson (TCTest2)
+ (void)load{
    NSLog(@"TCPerson (TCtest2) +load");
}
@end

可以看到我們在main里面不導入任何的頭文件身害,也不引用任何的類,直接運行侍瑟,控制臺輸出結(jié)果:


輸出結(jié)果

從輸出結(jié)果我們可以看出丙猬,三個load方法都被調(diào)用

問題:分類重寫方法,真的是覆蓋原有類的方法么庭瑰?如果不是弹灭,到底分類的方法是怎么調(diào)用的揪垄?

  • 首先我們在TCPerson申明一個方法+ (void)test并且在它的兩個分類都重寫+ (void)test

#import <Foundation/Foundation.h>

NS_ASSUME_NONNULL_BEGIN

@interface TCPerson : NSObject
+ (void)test;
@end

NS_ASSUME_NONNULL_END

#import "TCPerson.h"

@implementation TCPerson
+ (void)load{
    NSLog(@"TCPerson +load");
}
+ (void)test{
    NSLog(@"TCPerson +test");
}
@end

分類重寫test

#import "TCPerson+TCtest1.h"

@implementation TCPerson (TCtest1)
+ (void)load{
    NSLog(@"TCPerson (TCtest1) +load");
}
+ (void)test{
    NSLog(@"TCPerson (TCtest1) +test1");
}
@end
#import "TCPerson+TCTest2.h"

@implementation TCPerson (TCTest2)
+ (void)load{
    NSLog(@"TCPerson (TCtest2) +load");
}
+ (void)test{
    NSLog(@"TCPerson (TCtest2) +test2");
}
@end

在main里面我們調(diào)用test

#import <Foundation/Foundation.h>
#import "TCPerson.h"
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        [TCPerson test];
    }
    return 0;
}

輸出結(jié)果:

輸出結(jié)果

從輸出結(jié)果中我們可以看到饥努,只有分類2中的test被調(diào)用肪凛,為什么只調(diào)用分類2中的test了?


編譯順序

因為編譯順序是分類2在后翘鸭,1在前戳葵,這個時候我們改變編譯順序(拖動文件就行了)


改變后的順序

其輸出結(jié)果為:
image.png

細心的老鐵會看到拱烁,為什么load方法一直都在調(diào)用戏自,這是為什么了?它和test方法到底有什么不同了擅笔?真的是我們理解中的load不覆蓋,test覆蓋了念脯,所以才出現(xiàn)這種情況么绿店?

我們打印TCPerson的類方法

void printMethodNamesOfClass(Class cls)
{
    unsigned int count;
    // 獲得方法數(shù)組
    Method *methodList = class_copyMethodList(cls, &count);
    
    // 存儲方法名
    NSMutableString *methodNames = [NSMutableString string];
    
    // 遍歷所有的方法
    for (int i = 0; i < count; i++) {
        // 獲得方法
        Method method = methodList[I];
        // 獲得方法名
        NSString *methodName = NSStringFromSelector(method_getName(method));
        // 拼接方法名
        [methodNames appendString:methodName];
        [methodNames appendString:@", "];
    }
    
    // 釋放
    free(methodList);
    
    // 打印方法名
    NSLog(@"%@ %@", cls, methodNames);
}
int main(int argc, const char * argv[]) {
    @autoreleasepool {
        [TCPerson test];
        printMethodNamesOfClass(object_getClass([TCPerson class]));
    }
    return 0;
}

輸出結(jié)果:
打印

可以看到假勿,TCPerson的所有類方法名,并不是覆蓋,三個load堡距,三個test兆蕉,方法都在

load源碼分析:查看objc底層源碼我們可以看到:

void call_load_methods(void)
{
    static bool loading = NO;
    bool more_categories;

    loadMethodLock.assertLocked();

    // Re-entrant calls do nothing; the outermost call will finish the job.
    if (loading) return;
    loading = YES;

    void *pool = objc_autoreleasePoolPush();

    do {
        // 1. Repeatedly call class +loads until there aren't any more
        while (loadable_classes_used > 0) {
            call_class_loads();
        }

        // 2. Call category +loads ONCE
        more_categories = call_category_loads();

        // 3. Run more +loads if there are classes OR more untried categories
    } while (loadable_classes_used > 0  ||  more_categories);

    objc_autoreleasePoolPop(pool);

    loading = NO;
}

load方法它是先調(diào)用 while (loadable_classes_used > 0) {call_class_loads(); }類的load虎韵,再調(diào)用more_categories = call_category_loads()分類的load,和編譯順序無關(guān)驶社,都會調(diào)用
我們查看call_class_loads()方法

static void call_class_loads(void)
{
    int I;
    
    // Detach current loadable list.
    struct loadable_class *classes = loadable_classes;
    int used = loadable_classes_used;
    loadable_classes = nil;
    loadable_classes_allocated = 0;
    loadable_classes_used = 0;
    
    // Call all +loads for the detached list.
    for (i = 0; i < used; i++) {
        Class cls = classes[i].cls;
        load_method_t load_method = (load_method_t)classes[i].method;
        if (!cls) continue; 

        if (PrintLoading) {
            _objc_inform("LOAD: +[%s load]\n", cls->nameForLogging());
        }
        (*load_method)(cls, SEL_load);
    }
    
    // Destroy the detached list.
    if (classes) free(classes);
}

其通過的是load_method_t函數(shù)指針直接調(diào)用
函數(shù)指針直接調(diào)用

typedef void(*load_method_t)(id, SEL);

其分類load方法調(diào)用也是一樣

static bool call_category_loads(void)
{
    int i, shift;
    bool new_categories_added = NO;
    
    // Detach current loadable list.
    struct loadable_category *cats = loadable_categories;
    int used = loadable_categories_used;
    int allocated = loadable_categories_allocated;
    loadable_categories = nil;
    loadable_categories_allocated = 0;
    loadable_categories_used = 0;

    // Call all +loads for the detached list.
    for (i = 0; i < used; i++) {
        Category cat = cats[i].cat;
        load_method_t load_method = (load_method_t)cats[i].method;
        Class cls;
        if (!cat) continue;

        cls = _category_getClass(cat);
        if (cls  &&  cls->isLoadable()) {
            if (PrintLoading) {
                _objc_inform("LOAD: +[%s(%s) load]\n", 
                             cls->nameForLogging(), 
                             _category_getName(cat));
            }
            (*load_method)(cls, SEL_load);
            cats[i].cat = nil;
        }
    }

為什么test不一樣了

因為test是因為消息機制調(diào)用的亡电,objc_msgSend([TCPerson class], @selector(test));消息機制就牽扯到了isa方法查找份乒,test在元類方法里面順序查找的(關(guān)于isa腕唧,可以查看我的實例對象,類對象颂暇,元類對象的關(guān)聯(lián)---isa/superclass指針(2))里面有詳細的關(guān)于test的方法調(diào)用原理

load只在加載類的時候調(diào)用一次但惶,且先調(diào)用類的load瓣赂,再調(diào)用分類的

load的繼承關(guān)系調(diào)用
首先我們先看TCStudent

#import "TCStudent.h"

@implementation TCStudent

@end

不寫load方法調(diào)用


TCStudent不寫load

TCStudent寫上load


TCStudent寫上load

從中可以看出子類不寫load的方法,調(diào)用父類的load捌省,當子類調(diào)用load時,先調(diào)用父類的load卷拘,再調(diào)用子類的load祝高,父類子類load取決于你寫load方法沒有,如果都寫了乍赫,先調(diào)用父類的陆蟆,再調(diào)用子類的

總結(jié):先調(diào)用類的load叠殷,如果有子類,則先看子類是否寫了load像棘,如果寫了壶冒,則先調(diào)用父類的load,再調(diào)用子類的load避除,當類子類調(diào)用完了胸嘁,再是分類,分類的load取決于編譯順序群井,先編譯毫胜,則先調(diào)用诬辈,test的方法調(diào)用走的是消息發(fā)送機制焙糟,其底層原理和load方法有著本質(zhì)的區(qū)別样屠,消息發(fā)送主要取決于isa的方法查找順序

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市悦穿,隨后出現(xiàn)的幾起案子业踢,更是在濱河造成了極大的恐慌知举,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,591評論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異遮糖,居然都是意外死亡叠赐,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,448評論 3 392
  • 文/潘曉璐 我一進店門赛不,熙熙樓的掌柜王于貴愁眉苦臉地迎上來踢故,“玉大人惹苗,你說我怎么就攤上這事×芨伲” “怎么了院究?”我有些...
    開封第一講書人閱讀 162,823評論 0 353
  • 文/不壞的土叔 我叫張陵,是天一觀的道長伙窃。 經(jīng)常有香客問我对供,道長,這世上最難降的妖魔是什么鹅髓? 我笑而不...
    開封第一講書人閱讀 58,204評論 1 292
  • 正文 為了忘掉前任京景,我火速辦了婚禮确徙,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘鄙皇。我一直安慰自己伴逸,他們只是感情好,可當我...
    茶點故事閱讀 67,228評論 6 388
  • 文/花漫 我一把揭開白布洲愤。 她就那樣靜靜地躺著顷锰,像睡著了一般官紫。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上束世,一...
    開封第一講書人閱讀 51,190評論 1 299
  • 那天良狈,我揣著相機與錄音,去河邊找鬼遇西。 笑死,一個胖子當著我的面吹牛洲敢,可吹牛的內(nèi)容都是我干的茄蚯。 我是一名探鬼主播,決...
    沈念sama閱讀 40,078評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼皱碘!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起健蕊,我...
    開封第一講書人閱讀 38,923評論 0 274
  • 序言:老撾萬榮一對情侶失蹤缩功,失蹤者是張志新(化名)和其女友劉穎都办,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體世舰,經(jīng)...
    沈念sama閱讀 45,334評論 1 310
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,550評論 2 333
  • 正文 我和宋清朗相戀三年歼培,在試婚紗的時候發(fā)現(xiàn)自己被綠了茸塞。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,727評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖效扫,靈堂內(nèi)的尸體忽然破棺而出直砂,到底是詐尸還是另有隱情静暂,我是刑警寧澤谱秽,帶...
    沈念sama閱讀 35,428評論 5 343
  • 正文 年R本政府宣布疟赊,位于F島的核電站,受9級特大地震影響听绳,放射性物質(zhì)發(fā)生泄漏椅挣。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,022評論 3 326
  • 文/蒙蒙 一峡竣、第九天 我趴在偏房一處隱蔽的房頂上張望量九。 院中可真熱鬧,春花似錦类浪、人聲如沸肌似。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,672評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽眠蚂。三九已至斗躏,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間笛臣,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,826評論 1 269
  • 我被黑心中介騙來泰國打工碱鳞, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留踱蛀,地道東北人。 一個月前我還...
    沈念sama閱讀 47,734評論 2 368
  • 正文 我出身青樓崩泡,卻偏偏與公主長得像猬膨,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子谒所,可洞房花燭夜當晚...
    茶點故事閱讀 44,619評論 2 354

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