iOS runtime機制和使用

runtime簡稱運行時。OC是運行時機制,也就是在運行時才做一些處理。例如:C語言在編譯的時候就知道要調(diào)用哪個方法函數(shù),而OC在編譯的時候并不知道要調(diào)用哪個方法函數(shù),只有在運行的時候才知道調(diào)用的方法函數(shù)名稱,來找到對應(yīng)的方法函數(shù)進行調(diào)用。

導(dǎo)入

想要使用runtime,就要先導(dǎo)入runtime庫
一般導(dǎo)入message.h,因為message.h包含了objc.h和runtime.h

#import <objc/message.h>

runtime作用

一:發(fā)送消息
OC方法調(diào)用原理

方法調(diào)用的本質(zhì)就是放對象發(fā)送消息

/* new 會調(diào)用 init方法 */
People *man = [People new];
People *man = [[People alloc] init];

//屬性方法調(diào)用的方式
[man eat];
//類方法調(diào)用方式
[People eat];
[[People class] eat];

//還有一種不常用的調(diào)用方式
[對象/類 performSelector:@selector(eat)];

//底層實現(xiàn)
objc_msgSend(對象/屬性, @selector(eat));

最終代碼查看方法:clang -rewrite-objc main.m
消息發(fā)送底層實現(xiàn) Build Setting設(shè)置msg為NO Xcode5之后使用runtime機制
OC 與 C 的對應(yīng)方法
[People class] == objc_getClass("People")
@selector() == sel_registerName()

二:交換方法

當(dāng)系統(tǒng)自帶的方法功能不夠,可以給系統(tǒng)自帶的方法擴展一些功能,并保持原有的功能

例如我想知道當(dāng)前的URL是否為空如果每次都判斷一下的話會很麻煩,如果我創(chuàng)建擴展來寫,又不知道內(nèi)部是如何實現(xiàn)的.

一:使用繼承來實現(xiàn)

//.h文件內(nèi)容
#import <Foundation/Foundation.h>

@interface CFURL : NSURL

+(instancetype)CFURLWithString:(NSString *)string;

@end

-----------------------------------------------
//.m文件內(nèi)容
#import "CFURL.h"

@implementation CFURL

+(instancetype)CFURLWithString:(NSString *)string{
    CFURL *url = [super URLWithString:string];
    if (url == nil) {
        NSLog(@"url為空");
    }
    return url;
}

@end

二:使用runtime交換方法

//.h文件內(nèi)容
#import <Foundation/Foundation.h>

@interface NSURL (url)

+(instancetype)CF_URLWithStr:(NSString *)URLString;

@end

-----------------------------------------------
//.m文件內(nèi)容
#import "NSURL+url.h"

#import <objc/runtime.h>

@implementation NSURL (url)

+(void)load{
    //最早的方法,比main還早
    NSLog(@"load");
    //1.拿到兩個Method
    //2.進行方法交換
    Method m1 = class_getClassMethod([NSURL class], @selector(URLWithString:));
    Method m2 = class_getClassMethod([NSURL class], @selector(CF_URLWithStr:));
    //利用runtime進行方法的交換
    method_exchangeImplementations(m1, m2);
}


+(instancetype)CF_URLWithStr:(NSString *)URLString{
    //交換了兩個方法
    NSURL *url = [NSURL CF_URLWithStr:URLString];//注意這里不能再調(diào)用系統(tǒng)的方法
    if (!url) {
        NSLog(@"url為空");
    }
    return url;
}

@end

三:動態(tài)添加方法

//.m文件
#import "People.h"

#import <objc/runtime.h>

@implementation People

//當(dāng)類調(diào)用一個沒有實現(xiàn)的類方法就會到這里?。?+(BOOL)resolveClassMethod:(SEL)sel{
    NSLog(@"%@",NSStringFromSelector(sel));
    return [super resolveClassMethod:sel];
}

//當(dāng)類調(diào)用一個沒有實現(xiàn)的對象方法就會到這里?。?+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    if (sel == @selector(eat)) {
        // 動態(tài)添加eat方法
       /*
         第一個參數(shù):給哪個類添加方法
         第二個參數(shù):添加方法的方法編號
         第三個參數(shù):添加方法的函數(shù)實現(xiàn)(函數(shù)地址)
         第四個參數(shù):函數(shù)的類型,(返回值+參數(shù)類型) v:void @:對象->self :表示SEL->_cmd
       */
        class_addMethod(self, @selector(eat), eat, "v@:");
    }
    return [super resolveInstanceMethod:sel];
}

// 默認方法都有兩個隱式參數(shù),
void eat(id self,SEL sel)
{
    NSLog(@"%@ %@",self,NSStringFromSelector(sel));
}

@end

四: 給分類添加屬性

//.h文件
#import "NSObject.h"

@interface NSObject (Property)

//@property在分類中只會生成set、get方法的聲明 不會生成實現(xiàn),也不會生成_成員屬性
@property (nonatomic,copy)NSString name;

@end

-----------------------------------------------
//.m文件
// 定義關(guān)聯(lián)的key
static const char *key = "name";

@implementation NSObject (Property)

- (NSString *)name
{
    // 根據(jù)關(guān)聯(lián)的key,獲取關(guān)聯(lián)的值。
    return objc_getAssociatedObject(self, key);
}

- (void)setName:(NSString *)name
{
    /*
     第一個參數(shù):給哪個對象添加關(guān)聯(lián)
     第二個參數(shù):關(guān)聯(lián)的key,通過這個key獲取
     第三個參數(shù):關(guān)聯(lián)的value
     第四個參數(shù):關(guān)聯(lián)的策略
   */
    objc_setAssociatedObject(self, key, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

@end
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 2,074評論 0 9
  • 對于從事 iOS 開發(fā)人員來說,所有的人都會答出【runtime 是運行時】什么情況下用runtime?大部分人能...
    夢夜繁星閱讀 3,812評論 7 64
  • Runtime是什么 Runtime 又叫運行時,是一套底層的 C 語言 API,其為 iOS 內(nèi)部的核心之一,我...
    SuAdrenine閱讀 983評論 0 3
  • 寶寶也有自己的思想,他們說“不”可能是出于各種各樣的原因,這是成長中必經(jīng)的一步,我們作為爸爸媽媽的需要傾聽寶寶的心...
    君子如玉Anny閱讀 379評論 0 2
  • 這是我?guī)啄昵皩懙?,昨天偶爾發(fā)現(xiàn),細細看來,真的是一個‘樂’字在心頭。 時光流逝 過去的憂 ...
    Aes丨哀殉閱讀 133評論 0 1

友情鏈接更多精彩內(nèi)容