Objective-C中Category的本質(zhì)

寫(xiě)一個(gè) Person 的分類:Person+DO

Person+DO.h 文件:

#import "Person.h"

@interface Person (DO) <NSCopying>

@property (nonatomic, assign) int number;

- (void)testInstanceMethod;

+ (void)testClassMethod;

@end

Person_DO.m 文件:

#import "Person+DO.h"

@implementation Person (DO)

- (void)testInstanceMethod{}

+ (void)testClassMethod{}

@end

在終端輸入以下命令:

$ xcrun -sdk iphoneos clang -arch arm64 -rewrite-objc Person+DO.m -o Person+DO-arm64.cpp

打開(kāi) Person+DO-arm64.cpp 該文件贩毕,我們想要的都在這里面。

struct _category_t {
    const char *name;     // 類名
    struct _class_t *cls;
    const struct _method_list_t *instance_methods;     // 對(duì)象方法列表
    const struct _method_list_t *class_methods;    // 類方法列表
    const struct _protocol_list_t *protocols;     // 協(xié)議列表
    const struct _prop_list_t *properties;    // 屬性列表
};
static struct /*_method_list_t*/ {
    unsigned int entsize;  // sizeof(struct _objc_method)
    unsigned int method_count;
    struct _objc_method method_list[1];
} _OBJC_$_CATEGORY_INSTANCE_METHODS_Person_$_DO __attribute__ ((used, section ("__DATA,__objc_const"))) = {
    sizeof(_objc_method),
    1,
    {{(struct objc_selector *)"testInstanceMethod", "v16@0:8", (void *)_I_Person_DO_testInstanceMethod}}
};

_OBJC_$_CATEGORY_INSTANCE_METHODS_Person_$_DO :從名字(INSTANCE_METHODS)可看出是對(duì)象方法列表結(jié)構(gòu)體
testInstanceMethod:正是我們分類中定義的對(duì)象方法

static struct /*_method_list_t*/ {
    unsigned int entsize;  // sizeof(struct _objc_method)
    unsigned int method_count;
    struct _objc_method method_list[1];
} _OBJC_$_CATEGORY_CLASS_METHODS_Person_$_DO __attribute__ ((used, section ("__DATA,__objc_const"))) = {
    sizeof(_objc_method),
    1,
    {{(struct objc_selector *)"testClassMethod", "v16@0:8", (void *)_C_Person_DO_testClassMethod}}
};

_OBJC_$_CATEGORY_CLASS_METHODS_Person_$_DO :從名字(CLASS_METHODS)可看出是類方法列表結(jié)構(gòu)體
testClassMethod:正是我們分類中定義的類方法

static const char *_OBJC_PROTOCOL_METHOD_TYPES_NSCopying [] __attribute__ ((used, section ("__DATA,__objc_const"))) = 
{
    "@24@0:8^{_NSZone=}16"
};

static struct /*_method_list_t*/ {
    unsigned int entsize;  // sizeof(struct _objc_method)
    unsigned int method_count;
    struct _objc_method method_list[1];
} _OBJC_PROTOCOL_INSTANCE_METHODS_NSCopying __attribute__ ((used, section ("__DATA,__objc_const"))) = {
    sizeof(_objc_method),
    1,
    {{(struct objc_selector *)"copyWithZone:", "@24@0:8^{_NSZone=}16", 0}}
};

struct _protocol_t _OBJC_PROTOCOL_NSCopying __attribute__ ((used)) = {
    0,
    "NSCopying",
    0,
    (const struct method_list_t *)&_OBJC_PROTOCOL_INSTANCE_METHODS_NSCopying,
    0,
    0,
    0,
    0,
    sizeof(_protocol_t),
    0,
    (const char **)&_OBJC_PROTOCOL_METHOD_TYPES_NSCopying
};
struct _protocol_t *_OBJC_LABEL_PROTOCOL_$_NSCopying = &_OBJC_PROTOCOL_NSCopying;

static struct /*_protocol_list_t*/ {
    long protocol_count;  // Note, this is 32/64 bit
    struct _protocol_t *super_protocols[1];
} _OBJC_CATEGORY_PROTOCOLS_$_Person_$_DO __attribute__ ((used, section ("__DATA,__objc_const"))) = {
    1,
    &_OBJC_PROTOCOL_NSCopying
};

可看出,協(xié)議方法先通過(guò) _method_list_t 結(jié)構(gòu)體存儲(chǔ)凸克,之后通過(guò) _protocol_t 結(jié)構(gòu)體存儲(chǔ)在 _OBJC_CATEGORY_PROTOCOLS_$_Person_$_DO 中和 _protocol_list_t 結(jié)構(gòu)體一一對(duì)應(yīng)肾筐。

static struct /*_prop_list_t*/ {
    unsigned int entsize;  // sizeof(struct _prop_t)
    unsigned int count_of_properties;
    struct _prop_t prop_list[1];
} _OBJC_$_PROP_LIST_Person_$_DO __attribute__ ((used, section ("__DATA,__objc_const"))) = {
    sizeof(_prop_t),
    1,
    {{"number","Ti,N"}}
};

從屬性列表結(jié)構(gòu)體中慷吊,發(fā)現(xiàn)了我們自己寫(xiě)的 number 屬性

extern "C" __declspec(dllimport) struct _class_t OBJC_CLASS_$_Person;

static struct _category_t _OBJC_$_CATEGORY_Person_$_DO __attribute__ ((used, section ("__DATA,__objc_const"))) = 
{
    "Person",
    0, // &OBJC_CLASS_$_Person,
    (const struct _method_list_t *)&_OBJC_$_CATEGORY_INSTANCE_METHODS_Person_$_DO,
    (const struct _method_list_t *)&_OBJC_$_CATEGORY_CLASS_METHODS_Person_$_DO,
    (const struct _protocol_list_t *)&_OBJC_CATEGORY_PROTOCOLS_$_Person_$_DO,
    (const struct _prop_list_t *)&_OBJC_$_PROP_LIST_Person_$_DO,
};
static void OBJC_CATEGORY_SETUP_$_Person_$_DO(void ) {
    _OBJC_$_CATEGORY_Person_$_DO.cls = &OBJC_CLASS_$_Person;
}

struct _class_t OBJC_CLASS_$_Person_class_t 類型的 OBJC_CLASS_$_Person 結(jié)構(gòu)體
_OBJC_$_CATEGORY_Person_$_DO.cls = &OBJC_CLASS_$_Person;cls 指針指向分類的主類類對(duì)象 Person 的地址

至此馍惹,通過(guò)以上分類的源碼瑞眼,我們?cè)诜诸愔卸x的 對(duì)象方法 龙宏、 類方法屬性伤疙、協(xié)議 等都存放在 catagory_t 結(jié)構(gòu)體中银酗。

總結(jié):

  1. Category的實(shí)現(xiàn)原理:將分類中的方法,屬性徒像,協(xié)議數(shù)據(jù)放在 category_t 結(jié)構(gòu)體中黍特,然后將結(jié)構(gòu)體內(nèi)的方法列表拷貝到類對(duì)象的方法列表中。
  2. Category中不能加屬性:Category可以添加屬性锯蛀,但是并不會(huì)自動(dòng)生成成員變量及 set/get 方法灭衷。因?yàn)?category_t結(jié)構(gòu)體中并不存在成員變量。
    進(jìn)一步分析:
    成員變量是存放在實(shí)例對(duì)象中的旁涤,并且編譯使就已經(jīng)決定好了翔曲。而分類是在運(yùn)行時(shí)才去加載的,無(wú)法在程序運(yùn)行時(shí)將分類的成員變量添加到實(shí)例對(duì)象的結(jié)構(gòu)體中劈愚。因此分類中不可以添加成員變量瞳遍。

load 和 initialize

objc-loadmethod.mm 文件中:

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

優(yōu)先調(diào)用類的 load 方法,之后調(diào)用分類的 load 方法

void callInitialize(Class cls)
{
    ((void(*)(Class, SEL))objc_msgSend)(cls, SEL_initialize);
    asm("");
}

initialize 特點(diǎn):

  1. 第一次使用類的時(shí)候會(huì)調(diào)用(第一次接收到消息)
  2. 調(diào)用子類的 initialize 之前菌羽,會(huì)先保證調(diào)用父類的 initialize 方法
  3. 如果之前已經(jīng)調(diào)用過(guò) initialize掠械,就不會(huì)再調(diào)用 initialize 方法
  4. 當(dāng)分類重寫(xiě) initialize 方法時(shí)會(huì)先調(diào)用分類的方法
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);
}
// 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;
        }
    }

load 方法通過(guò)直接拿到 load 方法地址進(jìn)行調(diào)用

總結(jié):

loadinitialize 區(qū)別:

  1. load 是根據(jù)函數(shù)地址直接調(diào)用
    initialize 是通過(guò) objc_msgSend 調(diào)用
  2. loadruntime 加載類、分類的時(shí)候調(diào)用(只會(huì)調(diào)用1次)
    initialize 是類第一次接收到消息的時(shí)候調(diào)用算凿,每一個(gè)類只會(huì) initialize 一次(父類的 initialize 方法可能會(huì)被調(diào)用多次份蝴,子類未實(shí)現(xiàn))
  3. 分類中 load 方法不會(huì)覆蓋本類的 load 方法
    如果分類實(shí)現(xiàn)了 initialize 方法犁功,就覆蓋類本身的 initialize 方法
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末氓轰,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子浸卦,更是在濱河造成了極大的恐慌署鸡,老刑警劉巖,帶你破解...
    沈念sama閱讀 216,470評(píng)論 6 501
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異靴庆,居然都是意外死亡时捌,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,393評(píng)論 3 392
  • 文/潘曉璐 我一進(jìn)店門炉抒,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)奢讨,“玉大人,你說(shuō)我怎么就攤上這事焰薄∧弥睿” “怎么了?”我有些...
    開(kāi)封第一講書(shū)人閱讀 162,577評(píng)論 0 353
  • 文/不壞的土叔 我叫張陵塞茅,是天一觀的道長(zhǎng)亩码。 經(jīng)常有香客問(wèn)我,道長(zhǎng)野瘦,這世上最難降的妖魔是什么描沟? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 58,176評(píng)論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮鞭光,結(jié)果婚禮上吏廉,老公的妹妹穿的比我還像新娘。我一直安慰自己衰猛,他們只是感情好迟蜜,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,189評(píng)論 6 388
  • 文/花漫 我一把揭開(kāi)白布。 她就那樣靜靜地躺著啡省,像睡著了一般娜睛。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上卦睹,一...
    開(kāi)封第一講書(shū)人閱讀 51,155評(píng)論 1 299
  • 那天畦戒,我揣著相機(jī)與錄音,去河邊找鬼结序。 笑死障斋,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的徐鹤。 我是一名探鬼主播垃环,決...
    沈念sama閱讀 40,041評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼返敬!你這毒婦竟也來(lái)了遂庄?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 38,903評(píng)論 0 274
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤劲赠,失蹤者是張志新(化名)和其女友劉穎涛目,沒(méi)想到半個(gè)月后秸谢,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,319評(píng)論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡霹肝,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,539評(píng)論 2 332
  • 正文 我和宋清朗相戀三年估蹄,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片沫换。...
    茶點(diǎn)故事閱讀 39,703評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡臭蚁,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出讯赏,到底是詐尸還是另有隱情刊棕,我是刑警寧澤,帶...
    沈念sama閱讀 35,417評(píng)論 5 343
  • 正文 年R本政府宣布待逞,位于F島的核電站甥角,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏识樱。R本人自食惡果不足惜嗤无,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,013評(píng)論 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望怜庸。 院中可真熱鬧当犯,春花似錦、人聲如沸割疾。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 31,664評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)宏榕。三九已至拓诸,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間麻昼,已是汗流浹背奠支。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,818評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留抚芦,地道東北人倍谜。 一個(gè)月前我還...
    沈念sama閱讀 47,711評(píng)論 2 368
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像叉抡,于是被迫代替她去往敵國(guó)和親尔崔。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,601評(píng)論 2 353

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

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉褥民,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,709評(píng)論 0 9
  • iOS底層原理總結(jié) - Category的本質(zhì) 面試題 Category的實(shí)現(xiàn)原理季春,以及Category為什么只能...
    xx_cc閱讀 30,452評(píng)論 36 199
  • 1.ios高性能編程 (1).內(nèi)層 最小的內(nèi)層平均值和峰值(2).耗電量 高效的算法和數(shù)據(jù)結(jié)構(gòu)(3).初始化時(shí)...
    歐辰_OSR閱讀 29,372評(píng)論 8 265
  • OC語(yǔ)言基礎(chǔ) 1.類與對(duì)象 類方法 OC的類方法只有2種:靜態(tài)方法和實(shí)例方法兩種 在OC中,只要方法聲明在@int...
    奇異果好補(bǔ)閱讀 4,271評(píng)論 0 11
  • 新城煤礦是一座現(xiàn)代化煤礦轴捎,這個(gè)礦的創(chuàng)始人叫劉明善鹤盒,是一位具有非凡膽量的人,人稱“劉大膽”侦副,他的故事以及名氣侦锯,已經(jīng)蓋...
    東籬花飛閱讀 505評(píng)論 15 23