Objective-C中的類和對象

id和Class的定義

runtime里面灿巧,聲明了idClass的類型蹦肴,簡化一下如下:

struct objc_class {
    struct objc_class *isa;
};
struct objc_object {
    struct objc_class *isa;
};

typedef struct objc_class *Class;
typedef struct objc_object *id;

objc中僚碎,id代表了一個對象。根據(jù)上面的聲明阴幌,凡是首地址是*isastruct指針勺阐,都可以被認為是objc中的對象。運行時可以通過isa指針矛双,查找到該對象是屬于什么類(Class)渊抽。

運行時的實現(xiàn)方式

根據(jù)上面的說法,類對象(Class)同樣也算是對象议忽,那它的isa又是指向了什么呢懒闷?為了了解這些東西是怎么回事,這里寫一個簡單的類Cat,并且用C重寫一遍愤估,看看編譯器在底層到底是如何實現(xiàn)的帮辟。

//.h
@interface Cat : NSObject {
    int age;
    NSString *name;
}

- (void)nyan1;

+ (void)nyan2;

@end

//.m
@implementation Cat

- (void)nyan1 {
    printf("instance nyan~");
}

+ (void)nyan2 {
    printf("class nyan~");
}

@end

在終端執(zhí)行clang -rewrite-objc Cat.m這一條語句,讓clang將該類重寫為cpp代碼玩焰,我們就能查看到大概底層的實現(xiàn)機制了由驹。
rewrite后的代碼基本是純C的,稍微整理一下昔园,可以提取出下面這些信息:

//class 的實際結(jié)構(gòu)
struct _class_t {
    struct _class_t *isa;  //isa指針
    struct _class_t *superclass;  //父類
    void *cache;
    void *vtable;
    struct _class_ro_t *ro;  //class包含的信息
};

//Class包含的信息
struct _class_ro_t {
    unsigned int flags;
    unsigned int instanceStart;
    unsigned int instanceSize;
    unsigned int reserved;
    const unsigned char *ivarLayout;
    const char *name;  //類名
    const struct _method_list_t *baseMethods;  //方法列表
    const struct _objc_protocol_list *baseProtocols;  //協(xié)議列表
    const struct _ivar_list_t *ivars;  //ivar列表
    const unsigned char *weakIvarLayout;
    const struct _prop_list_t *properties;  //屬性列表
};

extern "C" __declspec(dllimport) struct _class_t OBJC_METACLASS_$_NSObject;

extern "C" __declspec(dllexport) struct _class_t OBJC_METACLASS_$_Cat __attribute__ ((used, section ("__DATA,__objc_data"))) = {
    0, // &OBJC_METACLASS_$_NSObject,
    0, // &OBJC_METACLASS_$_NSObject,
    0, // (void *)&_objc_empty_cache,
    0, // unused, was (void *)&_objc_empty_vtable,
    &_OBJC_METACLASS_RO_$_Cat,  //包含了類方法等
};

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

extern "C" __declspec(dllexport) struct _class_t OBJC_CLASS_$_Cat __attribute__ ((used, section ("__DATA,__objc_data"))) = {
    0, // &OBJC_METACLASS_$_Cat,  //此處isa指向meta-class
    0, // &OBJC_CLASS_$_NSObject,
    0, // (void *)&_objc_empty_cache,
    0, // unused, was (void *)&_objc_empty_vtable,
    &_OBJC_CLASS_RO_$_Cat,  //包含了實例方法 ivar信息等
};
typedef struct objc_object Cat;

1蔓榄、所有Cat的實例的isa都指向了Cat(Class);
2、Cat(Class)是一個全局變量蒿赢,其中記錄了類名润樱、成員變量信息、property信息羡棵、protocol信息和實例方法列表等;
3壹若、Cat(Class)isa指向了全局變量Cat(meta-class)meta-class里只記錄了類名皂冰、類方法列表等店展。

objc-meta.png

類的繼承

為了說明方便,這里把上面的例子稍微改一下:NyanCat : Cat : NSObject 這樣一個繼承樹秃流,畫出圖來就是這樣子的:

image

可變與不可變

因為對象在內(nèi)存中的排布可以看成一個結(jié)構(gòu)體赂蕴,該結(jié)構(gòu)體的大小并不能動態(tài)變化。所以無法在運行時給對象添加成員變量舶胀。
但同時在 objc_class的結(jié)構(gòu)體中概说,我們可以發(fā)現(xiàn)方法的定義列表時一個名為methodLists的指針的指針,通過修改該指針所指向的指針的值,就可以動態(tài)地為某一個類增加成員方法嚣伐。這也是Category實現(xiàn)的原理糖赔。同時也說明了為什么Category只可為對象增加成員方法,卻不能增加成員變量轩端。

需要特別說明一下放典,通過objc_setAssociatedObject 和 objc_getAssociatedObject方法可以變相地給對象增加成員變量,但由于實現(xiàn)機制不一樣基茵,所以并不是真正改變了對象的內(nèi)存結(jié)構(gòu)奋构。

系統(tǒng)相關(guān) API 及應(yīng)用

Objective-C 提供了以下 API 來動態(tài)替換類方法或?qū)嵗椒ǖ膶崿F(xiàn):

  • class_replaceMethod 替換類方法的定義
  • method_exchangeImplementations 交換 2 個方法的實現(xiàn)
  • method_setImplementation 設(shè)置 1 個方法的實現(xiàn)

使用場景:

  • class_replaceMethod, 當需要替換的方法可能有不存在的情況時,可以考慮使用該方法拱层。
  • method_exchangeImplementations弥臼,當需要交換 2 個方法的實現(xiàn)時使用。
  • method_setImplementation 最簡單的用法舱呻,當僅僅需要為一個方法設(shè)置其實現(xiàn)方式時使用醋火。

使用舉例:

// ImagePickerReplaceMethodsHolder.h
@interface ImagePickerReplaceMethodsHolder : NSObject
- (BOOL)shouldAutorotate;
- (UIInterfaceOrientation)preferredInterfaceOrientationForPresentation;
@end
// ImagePickerReplaceMethodsHolder.m
@implementation ImagePickerReplaceMethodsHolder
- (BOOL)shouldAutorotate {
    return NO;
}
- (UIInterfaceOrientation)preferredInterfaceOrientationForPresentation {
    return UIInterfaceOrientationPortrait;
}
@end
#define SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(v)  ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] != NSOrderedAscending)
#define SYSTEM_VERSION_LESS_THAN(v)                 ([[[UIDevice currentDevice] systemVersion] compare:v options:NSNumericSearch] == NSOrderedAscending)
+ (void)load {
    static dispatch_once_t onceToken;
    dispatch_once(&onceToken, ^{
        [self hackForImagePicker];
    });
}
+ (void)hackForImagePicker {
    // fix bug of image picker under iOS 6.0
    // http://stackoverflow.com/questions/12522491/crash-on-presenting-uiimagepickercontroller-under-ios-6-0
    if (SYSTEM_VERSION_GREATER_THAN_OR_EQUAL_TO(@"6.0")
        && SYSTEM_VERSION_LESS_THAN(@"6.1")) {
        Method oldMethod1 = class_getInstanceMethod([UIImagePickerController class], @selector(shouldAutorotate));
        Method newMethod1 = class_getInstanceMethod([ImagePickerReplaceMethodsHolder class], @selector(shouldAutorotate));
        method_setImplementation(oldMethod1, method_getImplementation(newMethod1));
        Method oldMethod2 = class_getInstanceMethod([UIImagePickerController class], @selector(preferredInterfaceOrientationForPresentation));
        Method newMethod2 = class_getInstanceMethod([ImagePickerReplaceMethodsHolder class], @selector(preferredInterfaceOrientationForPresentation));
        method_setImplementation(oldMethod2, method_getImplementation(newMethod2));
    }
}

KVO和KVC的實現(xiàn)原理

  • KVO

當你觀察一個對象時悠汽,一個新的類會被動態(tài)創(chuàng)建。這個類繼承自該對象的原本的類芥驳,并重寫了被觀察屬性的 setter 方法柿冲。重寫的 setter 方法會負責在調(diào)用原 setter 方法之前和之后,通知所有觀察對象:值的更改兆旬。最后通過isa 混寫isa-swizzling) 把這個對象的 isa 指針 ( isa 指針告訴 Runtime 系統(tǒng)這個對象的類是什么 ) 指向這個新創(chuàng)建的子類假抄,對象就神奇的變成了新創(chuàng)建的子類的實例。我畫了一張示意圖丽猬,如下所示:

KVC 支持實例變量宿饱,KVO 只能手動支持手動設(shè)定實例變量的KVO實現(xiàn)監(jiān)聽

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市脚祟,隨后出現(xiàn)的幾起案子谬以,更是在濱河造成了極大的恐慌,老刑警劉巖由桌,帶你破解...
    沈念sama閱讀 222,104評論 6 515
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件为黎,死亡現(xiàn)場離奇詭異,居然都是意外死亡行您,警方通過查閱死者的電腦和手機铭乾,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,816評論 3 399
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來娃循,“玉大人炕檩,你說我怎么就攤上這事“聘” “怎么了笛质?”我有些...
    開封第一講書人閱讀 168,697評論 0 360
  • 文/不壞的土叔 我叫張陵,是天一觀的道長捞蚂。 經(jīng)常有香客問我经瓷,道長,這世上最難降的妖魔是什么洞难? 我笑而不...
    開封第一講書人閱讀 59,836評論 1 298
  • 正文 為了忘掉前任,我火速辦了婚禮揭朝,結(jié)果婚禮上队贱,老公的妹妹穿的比我還像新娘。我一直安慰自己潭袱,他們只是感情好柱嫌,可當我...
    茶點故事閱讀 68,851評論 6 397
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著屯换,像睡著了一般编丘。 火紅的嫁衣襯著肌膚如雪与学。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,441評論 1 310
  • 那天嘉抓,我揣著相機與錄音索守,去河邊找鬼。 笑死抑片,一個胖子當著我的面吹牛卵佛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播敞斋,決...
    沈念sama閱讀 40,992評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼截汪,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了植捎?” 一聲冷哼從身側(cè)響起繁仁,我...
    開封第一講書人閱讀 39,899評論 0 276
  • 序言:老撾萬榮一對情侶失蹤嘱能,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體昧识,經(jīng)...
    沈念sama閱讀 46,457評論 1 318
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,529評論 3 341
  • 正文 我和宋清朗相戀三年搪搏,在試婚紗的時候發(fā)現(xiàn)自己被綠了灰追。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,664評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡拟淮,死狀恐怖干茉,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情很泊,我是刑警寧澤角虫,帶...
    沈念sama閱讀 36,346評論 5 350
  • 正文 年R本政府宣布,位于F島的核電站委造,受9級特大地震影響戳鹅,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜昏兆,卻給世界環(huán)境...
    茶點故事閱讀 42,025評論 3 334
  • 文/蒙蒙 一枫虏、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧爬虱,春花似錦隶债、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,511評論 0 24
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至曲梗,卻和暖如春赞警,著一層夾襖步出監(jiān)牢的瞬間妓忍,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,611評論 1 272
  • 我被黑心中介騙來泰國打工愧旦, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留世剖,地道東北人。 一個月前我還...
    沈念sama閱讀 49,081評論 3 377
  • 正文 我出身青樓忘瓦,卻偏偏與公主長得像搁廓,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子耕皮,可洞房花燭夜當晚...
    茶點故事閱讀 45,675評論 2 359

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