runtime詳解內(nèi)容補充

最近研究了一下oc底層的runtime機制秤标,在網(wǎng)上找到了一篇不錯的文章對于runtime講的也比較詳細(iOS Runtime詳解)。
對于runtime不太了解的同學(xué)可以先看下這篇文章睛挚,下面是一些我的理解和內(nèi)容的補充。

關(guān)于元類

根據(jù)基類NSObject我們知道在每個類中都有一個isa偎捎,實際上是一個objc_class的結(jié)構(gòu)體(runtime.h中有顯示)婴程。
其中有兩個字段是指向其元類的isa和指向父類的supperclass(兩者都是Class也就是objc_class)。


結(jié)構(gòu)關(guān)系

在objc_class中有一個methodLists字段存放的是類的方法渠脉,通過實驗(class_copyMethodList獲取這個方法列表)宇整,我發(fā)現(xiàn)類中的methodLists都是實例方法,而元類都是類方法芋膘。
所以元類是存儲了這個類的類方法鳞青,并且保證了子類也能調(diào)用到父類的類方法(因為類的元類的父類指向的是類的父類的元類)。

關(guān)于消息轉(zhuǎn)發(fā)機制的應(yīng)用(交換兩個方法)

這邊大多數(shù)主要是為了全局修改某個方法實現(xiàn)为朋。
第一種方法臂拓,比如給NSMutableDictionary的setObject:forKey:方法添加nil判斷,建一個NSMutableDictionary的分類添加下面方法:

+ (void)load {
    [self changeMehtod];
}
+ (void)changeMehtod {
    //這邊注意不用[NSMutableDictionary class]是因為NSMutableDictionary屬于類簇
    NSMutableDictionary *dict = [NSMutableDictionary new];
    Method originMethod = class_getInstanceMethod(object_getClass(dict), @selector(setObject:forKey:));
    Method targetMethod = class_getInstanceMethod(object_getClass(dict), @selector(handleSetObject:forKey:));
    method_exchangeImplementations(originMethod, targetMethod);
}

- (void)handleSetObject:(id)anobject forKey:(id)akey {
    if (anobject == nil) {
        anobject = @"error";
    }
    //因為交換了方法的關(guān)系习寸,系統(tǒng)的setObject:forKey:
    [self handleSetObject:anobject forKey:akey]; 
}

還有通過block的方式胶惰,比如給UIViewController要在執(zhí)行ViewDidLoad前執(zhí)行一些操作:

+ (void)load {
    [self changeMehtod];
}
+ (void)changeMehtod{
   Method method = class_getInstanceMethod([UIViewController class], @selector(viewDidLoad));
    
    void (*originIMP)(id self, SEL _cmd) = nil;
    originIMP = method_getImplementation(method);
    
    id newViewDidLoad = ^(id self, SEL _cmd){
        NSLog(@"operation");
        originIMP(self, _cmd);
    };
    
    IMP newIMP = imp_implementationWithBlock(newViewDidLoad);
    method_setImplementation(method, newIMP);
}

我們知道oc的方法調(diào)用本質(zhì)上就是消息轉(zhuǎn)發(fā)
在一個方法Method中有一個SEL和IMP,SEL是方法的編號實際上就是一個char類型數(shù)組字符串霞溪,比如viewDidLoad方法的SEL打印出來就是"viewDidLoad"這個在沒運行程序時就能確定孵滞,而動態(tài)變的是IMP是方法的具體實現(xiàn),還有一個SEL和IMP的對應(yīng)表鸯匹。方法交換本質(zhì)上就是交換SEL和IMP的對應(yīng)關(guān)系坊饶。

- (void)methodIMP{
//相當(dāng)于[self testMethod:@"123"];
    ((void (*)(id ,SEL, NSString*))objc_msgSend)(self, @selector(testMethod:), @"123");
}

- (void)testMethod:(NSString*)agr{  
    NSLog(@"%@", agr);
}

關(guān)于分類

對于分類為什么能添加方法或者說覆蓋原來類中方法名相同的方法呢?
我做了一個實驗忽你,新建一個方法,寫了一個方法hello臂容,并建了一個它的分類也實現(xiàn)了這個科雳,方法然后實例話類調(diào)用方法根蟹,這時候調(diào)用的分類的方法。
然后我查看了這個類的方法列表methodLists糟秘,發(fā)現(xiàn)有兩個名字都是hello方法简逮。

    [[TestFunctionOne new] hello];

    int outCount;
    Method *methods = class_copyMethodList([TestFunctionOne class], &outCount);
    for (int i = 0; i < outCount; i++) {
        Method method = methods[i];
        NSLog(@"%s",sel_getName(method_getName(method)));
    }

從這里可以看出category實際上就是把方法插到了類中的methodLists中的前面,在調(diào)用方法轉(zhuǎn)發(fā)消息的時候尿赚,類通過methodLists依次尋找方法散庶,結(jié)果先找到了分類的方法。

關(guān)于類的動態(tài)創(chuàng)建

由于runtime機制的動態(tài)性凌净,我們可以通過在代碼運行時創(chuàng)建一個類(雖然具體使用場景我沒有想到..)悲龟,具體如下:

+ (void)createClass{
    //TestClass判斷這個類是否存在
    if (NSClassFromString(@"TestClass")) {
        return;
    }
    
    // 1 創(chuàng)建
    Class EOCTestClass = objc_allocateClassPair([NSObject class], "TestClass", 0);
    
    // 2 添加變量
    if(class_addIvar(TestClass, "str", sizeof(NSString*), log2(sizeof(NSString*)), @encode(NSString*))){
        NSLog(@"添加成功");
    }
    
    if(class_addIvar(EOCTestClass, "intNum", sizeof(int), log2(sizeof(int)), @encode(int))){
        NSLog(@"添加成功");
    }
    
    // 3 提交 (提交完之后, 就不能在添加變量)
    objc_registerClassPair(TestClass);
    
    // 4 測試
    id testObject = [TestClass new];
    
    [testObject setValue:@"string" forKey:@"str"];
    NSLog(@"%@", [testObject valueForKey:@"str"]);    
}

就先寫這些冰寻,之后有新的體會在補充 :)

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末须教,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子斩芭,更是在濱河造成了極大的恐慌轻腺,老刑警劉巖,帶你破解...
    沈念sama閱讀 218,755評論 6 507
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件划乖,死亡現(xiàn)場離奇詭異贬养,居然都是意外死亡,警方通過查閱死者的電腦和手機琴庵,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,305評論 3 395
  • 文/潘曉璐 我一進店門误算,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人细卧,你說我怎么就攤上這事尉桩。” “怎么了贪庙?”我有些...
    開封第一講書人閱讀 165,138評論 0 355
  • 文/不壞的土叔 我叫張陵蜘犁,是天一觀的道長。 經(jīng)常有香客問我止邮,道長这橙,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,791評論 1 295
  • 正文 為了忘掉前任导披,我火速辦了婚禮屈扎,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘撩匕。我一直安慰自己鹰晨,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,794評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著模蜡,像睡著了一般漠趁。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上忍疾,一...
    開封第一講書人閱讀 51,631評論 1 305
  • 那天闯传,我揣著相機與錄音,去河邊找鬼卤妒。 笑死甥绿,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的则披。 我是一名探鬼主播共缕,決...
    沈念sama閱讀 40,362評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼收叶!你這毒婦竟也來了骄呼?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,264評論 0 276
  • 序言:老撾萬榮一對情侶失蹤判没,失蹤者是張志新(化名)和其女友劉穎蜓萄,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體澄峰,經(jīng)...
    沈念sama閱讀 45,724評論 1 315
  • 正文 獨居荒郊野嶺守林人離奇死亡嫉沽,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,900評論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了俏竞。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片绸硕。...
    茶點故事閱讀 40,040評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖魂毁,靈堂內(nèi)的尸體忽然破棺而出玻佩,到底是詐尸還是另有隱情,我是刑警寧澤席楚,帶...
    沈念sama閱讀 35,742評論 5 346
  • 正文 年R本政府宣布咬崔,位于F島的核電站,受9級特大地震影響烦秩,放射性物質(zhì)發(fā)生泄漏垮斯。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 41,364評論 3 330
  • 文/蒙蒙 一只祠、第九天 我趴在偏房一處隱蔽的房頂上張望兜蠕。 院中可真熱鬧,春花似錦抛寝、人聲如沸熊杨。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,944評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽晶府。三九已至夷狰,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間郊霎,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,060評論 1 270
  • 我被黑心中介騙來泰國打工爷绘, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留书劝,地道東北人。 一個月前我還...
    沈念sama閱讀 48,247評論 3 371
  • 正文 我出身青樓土至,卻偏偏與公主長得像购对,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子陶因,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,979評論 2 355

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

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉骡苞,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,719評論 0 9
  • 引導(dǎo) 對于從事 iOS 開發(fā)人員來說,所有的人都會答出「 Runtime 是運行時 」楷扬,什么情況下用 Runtim...
    Winny_園球閱讀 4,209評論 3 75
  • 這篇文章完全是基于南峰子老師博客的轉(zhuǎn)載 這篇文章完全是基于南峰子老師博客的轉(zhuǎn)載 這篇文章完全是基于南峰子老師博客的...
    西木閱讀 30,560評論 33 466
  • 對于從事 iOS 開發(fā)人員來說解幽,所有的人都會答出【runtime 是運行時】什么情況下用runtime?大部分人能...
    夢夜繁星閱讀 3,721評論 7 64
  • 我們常常會聽說 Objective-C 是一門動態(tài)語言,那么這個「動態(tài)」表現(xiàn)在哪呢烘苹?我想最主要的表現(xiàn)就是 Obje...
    Ethan_Struggle閱讀 2,195評論 0 7