Objective-C Runtime介紹與應(yīng)用示例

在看一些牛逼閃閃的開源框架時發(fā)現(xiàn)都使用了Runtime黑魔法,那么究竟什么是Runtime呢?我們都知道Objective-C是一門動態(tài)語言,Objective-C最大的特色是承自Smalltalk的消息傳遞模型(message passing),這種機(jī)制和當(dāng)今C++式的主流風(fēng)格差異甚大,C++里類別與方法的關(guān)系嚴(yán)格清楚,而在Objective-C中,類別與消息的關(guān)系比較松散,調(diào)用方法視為對對象發(fā)送消息,所有方法都被視為對消息的回應(yīng)。所有消息處理直到運(yùn)行時(runtime)才會動態(tài)決定,并交由類別自行決定如何處理收到的消息。簡單地說,Runtime系統(tǒng)是一個包含由一系列函數(shù)和數(shù)據(jù)結(jié)構(gòu)組成公共接口的動態(tài)共享庫,在頭文件位于/usr/include/objc.的目錄。當(dāng)你編寫objective - C代碼的時候,這些函數(shù)允許您使用純C代碼去復(fù)制,并在編譯時執(zhí)行這些函數(shù)。

Objective-C程序與runtime system的交互體現(xiàn)在三個不同的層次:
1.通過Objective-C源代碼;
2.通過Foundation frameworkNSObject類定義的方法;
3.通過直接調(diào)用runtime函數(shù)。

一:Runtime最主要的就是消息機(jī)制。我們從The objc_msgSend Function——消息發(fā)送開始說起。

[receiver message]

objective - c中,直到運(yùn)行時消息才會和對應(yīng)實(shí)現(xiàn)的方法綁定。消息調(diào)用時的轉(zhuǎn)換是在編譯期間進(jìn)行的。上面的代碼實(shí)際上會被編譯器轉(zhuǎn)化為:

objc_msgSend(receiver, selector)

objc_msgSend是一個消息發(fā)送傳遞的函數(shù),。這個函數(shù)需要消息接收者和消息方法名(這個方法名選擇器) 作為它的兩個主要參數(shù)。

objc_msgSend(receiver, selector, arg1, arg2, ...)

消息傳入的任何參數(shù)也會交給objc_msgSend。

objc/message.h文件中可以看到objc_msgSend 函數(shù)的定義:

OBJC_EXPORT id objc_msgSend(id self, SEL op, ...)

SEL
objc_msgSend函數(shù)第二個參數(shù)類型為SEL,它是selectorOC中的表示類型。selector是方法選擇器,可以理解為區(qū)分方法的ID,而這個 ID的數(shù)據(jù)結(jié)構(gòu)是SEL:它被定義在objc/objc.h目錄下:

/// An opaque type that represents a method selector.
typedef struct objc_selector *SEL;

id
objc_msgSend函數(shù)第一個參數(shù)類型為id,與SEL 一樣,id也被定義在 objc/objc.h 目錄下:

/// A pointer to an instance of a class.
typedef struct objc_object *id;

id 是一個結(jié)構(gòu)體指針類型,它可以指向 Objective-C中的任何對象。objc_object結(jié)構(gòu)體定義如下:

/// Represents an instance of a class.
struct objc_object {
    Class isa  OBJC_ISA_AVAILABILITY;
};

objc_object結(jié)構(gòu)體包含一個isa指針,根據(jù)isa指針就可以順藤摸瓜找到對象所屬的類。

Class
之所以說isa是指針是因為Class其實(shí)是一個指向objc_class結(jié)構(gòu)體的指針:

/// An opaque type that represents an Objective-C class.
typedef struct objc_class *Class;

那么objc_class的廬山真面目又是什么呢?進(jìn)入objc/runtime.h我們便可一窺究竟:

struct objc_class {
    Class isa  OBJC_ISA_AVAILABILITY;

#if !__OBJC2__
    Class super_class                                        OBJC2_UNAVAILABLE;
    const char *name                                         OBJC2_UNAVAILABLE;
    long version                                             OBJC2_UNAVAILABLE;
    long info                                                OBJC2_UNAVAILABLE;
    long instance_size                                       OBJC2_UNAVAILABLE;
    struct objc_ivar_list *ivars                             OBJC2_UNAVAILABLE;
    struct objc_method_list **methodLists                    OBJC2_UNAVAILABLE;
    struct objc_cache *cache                                 OBJC2_UNAVAILABLE;
    struct objc_protocol_list *protocols                     OBJC2_UNAVAILABLE;
#endif

} OBJC2_UNAVAILABLE;

這就是我們常說的類:
·Class 也有一個 isa 指針,指向其所屬的元類(meta).
·super_class:指向其超類.
·name:是類名.
·version:是類的版本信息.
·info:類的詳情.
·instance_size:是該類的實(shí)例對象的大小.
·ivars:指向該類的成員變量列表.
·methodLists:指向該類的實(shí)例方法列表,它將方法選擇器和方法實(shí)現(xiàn)地址聯(lián)系起來。methodLists 是指向 ·objc_method_list 指針的指針,也就是說可以動態(tài)修改 *methodLists 的值來添加成員方法,這也是Category實(shí)現(xiàn)的原理,同樣解釋了Category不能添加屬性的原因.
·cache:Runtime系統(tǒng)會把被調(diào)用的方法存到 cache中(理論上講一個方法如果被調(diào)用,那么它有可能今后還會被調(diào)用),下次查找的時候效率更高.
·protocols:指向該類的協(xié)議列表.

總結(jié)一下:
首先,Runtime 系統(tǒng)會把方法調(diào)用轉(zhuǎn)化為消息發(fā)送,即 objc_msgSend,并且把方法的調(diào)用者,和方法選擇器,當(dāng)做參數(shù)傳遞過去.此時,方法的調(diào)用者會通過isa 指針來找到其所屬的類,然后在cache 或者methodLists 中查找該方法,找得到就跳到對應(yīng)的方法去執(zhí)行.
如果在類中沒有找到該方法,則通過super_class往上一級超類查找(如果一直找到NSObject都沒有找到該方法的話,就會調(diào)用下面這幾個方法,給你“補(bǔ)救”的機(jī)會,你可以先理解為幾套防止程序crash的備選方案,我們就是利用這幾個方案進(jìn)行消息轉(zhuǎn)發(fā),注意一點(diǎn),前一套方案實(shí)現(xiàn)后一套方法就不會執(zhí)行。如果這幾套方案你都沒有做處理,那么程序就會報錯crash

方案一:
+ (BOOL)resolveInstanceMethod:(SEL)sel
+ (BOOL)resolveClassMethod:(SEL)sel
方案二:
- (id)forwardingTargetForSelector:(SEL)aSelector

方案三:

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector;
- (void)forwardInvocation:(NSInvocation *)anInvocation;

示意圖:

前面我們說 methodLists指向該類的實(shí)例方法列表,實(shí)例方法即-方法,那么類方法(+方法)存儲在哪兒呢?類方法被存儲在元類中,Class通過 isa 指針即可找到其所屬的元類。

上圖實(shí)線是 super_class 指針,虛線是 isa 指針。根元類的超類是NSObject,而 isa 指向了自己。NSObject 的超類為 nil,也就是它沒有超類。

介紹完理論,下面讓我們開始真正在開發(fā)中使用objc_msgSend吧!
1.創(chuàng)建Fish類并添加方法:

@interface Fish : NSObject
//游泳
+ (void)swim;
- (void)swim;

//吐泡泡
- (void)bloweBubbles:(int)num;
@end

2.使用objc_msgSend

//類對象發(fā)送消息
- (void)testClassObject{
    //獲取類對象
    Class fClass = [Fish class];
    //運(yùn)行時
    objc_msgSend(fClass, @selector(bloweBubbles:),100);
}

//對象發(fā)送消息
- (void)testObject{
    
    Fish * fish = [[Fish alloc]init];
    //    運(yùn)行時,發(fā)送消息
    objc_msgSend(fish, @selector(swim));
    
    //    帶參數(shù)
    objc_msgSend(fish, @selector(bloweBubbles:),10);
}

二:Method Swizzling
Method Swizzing是發(fā)生在運(yùn)行時的,主要用于在運(yùn)行時將兩個方法進(jìn)行交換。當(dāng)我們在開發(fā)中發(fā)現(xiàn)系統(tǒng)自帶的方法功能不夠,需要給系統(tǒng)自帶的方法擴(kuò)展一些功能,并且保持原有的功能時,Method Swizzing便派上用場了。
用法示例:比如我們想給UIImageimageNamed:方法增加圖片加載是否成功的提示。

1.創(chuàng)建UIImage的分類并聲明作為交換的方法:

#import <UIKit/UIKit.h>

@interface UIImage (JF)

+ (__kindof UIImage *)imageWithName:(NSString *)name;

@end

2.實(shí)現(xiàn)Method Swizzing

#import "UIImage+JF.h"
#import <objc/message.h>
@implementation UIImage (JF)
+ (void)load
{
   // 交換方法
    // 獲取imageWithName方法
    Method imageWithName = class_getClassMethod(self, @selector(imageWithName:));
    
    // 獲取imageNamed方法
    Method imageNamed = class_getClassMethod(self, @selector(imageNamed:));
    
    // 交換方法,相當(dāng)于交換函數(shù)地址
    method_exchangeImplementations(imageWithName, imageNamed);
    
}

// 既能加載圖片又能提示是否加載成功
+ (__kindof UIImage *)imageWithName:(NSString *)name
{
  // 這里調(diào)用imageWithName,相當(dāng)于調(diào)用imageName
    UIImage * image = [self imageWithName:name];
  //用打印模擬提示功能
    if (image == nil) {
        NSLog(@"圖片加載失敗");
    }
    return image;
}

三:動態(tài)方法處理
有時候,我們可能需要提供一個動態(tài)的實(shí)現(xiàn)方法,具體如下:
創(chuàng)建Fish類,動態(tài)添加drink:方法

#import "Fish.h"
@implementation Fish

//定義函數(shù)
//沒有返回值,參數(shù)(id,SEL,id)
//void(id,SEL,id)
void drink(id self,SEL _cmd,id param1)
{
    NSLog(@"魚兒%@,%@,%@口水",self,NSStringFromSelector(_cmd),param1);
}


//動態(tài)添加方法首先實(shí)現(xiàn)resolveInstanceMethod
/*
 resolveInstanceMethod調(diào)用:當(dāng)調(diào)用了沒有實(shí)現(xiàn)的方法,就會調(diào)用該方法resolveInstanceMethod
 resolveInstanceMethod的作用:知道哪些方法沒有實(shí)現(xiàn),從而動態(tài)添加方法
 sel:沒有實(shí)現(xiàn)的方法的編碼
 */
+ (BOOL)resolveInstanceMethod:(SEL)sel{
    
    //動態(tài)添加drink方法
    if( sel == @selector(drink:))
    {
        /*
         cls:給哪個類添加方法
         SEl:添加方法的方法編號
         IMP:方法實(shí)現(xiàn),函數(shù)入口,函數(shù)名
         types:方法類型
         */
        class_addMethod(self, sel, (IMP)drink, "v@:@");
        
    }
        
        return YES;
}
- (void)viewDidLoad {
    [super viewDidLoad];

    Fish * f = [[Fish alloc]init];
    [f performSelector:@selector(drink:) withObject:@10];
}

** 四:動態(tài)添加屬性 **
在一般情況下,我們知道在分類中是無法添加屬性的,因為@property在分類中,只會生成get,set方法的聲明,不會生成下劃線成員屬性,和get,set方法的實(shí)現(xiàn)。但是,通過Runtime我們可以實(shí)現(xiàn)動態(tài)地添加屬性。

1.創(chuàng)建NSObject分類

#import <Foundation/Foundation.h>
@interface NSObject (JF)

@property(nonatomic,strong)NSString * name;

@end

#import "NSObject+JF.h"
#import <objc/message.h>

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

@implementation NSObject (JF)

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

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

** 五:Runtime字典轉(zhuǎn)模型 **
1.遍歷模型中所有屬性
2.給模型中的每個屬性賦值

#import "NSObject+JFExtension.h"
#import <objc/message.h>

@implementation NSObject (JFExtension)

+ (instancetype)modelWithDic:(NSDictionary *)dic{
    
    // 思路:遍歷模型中所有屬性-》使用運(yùn)行時
    // 0.創(chuàng)建對應(yīng)的對象
    id objc = [[self alloc]init];
    
    // 1.利用runtime給對象中的成員屬性賦值
    unsigned int count;
    // 獲取類中的所有成員屬性
    Ivar * ivarList = class_copyIvarList(self, &count);
    
    for (int i = 0; i < count; i++) {
        // 根據(jù)角標(biāo),從數(shù)組取出對應(yīng)的成員屬性
        Ivar ivar = ivarList[i];
        
        // 獲取成員屬性名
        NSString * propertyName = [NSString stringWithUTF8String:ivar_getName(ivar)];
        
        // 處理成員屬性名->字典中的key
        // 從第一個角標(biāo)開始截取
        NSString * key = [propertyName substringFromIndex:1];
        
        // 根據(jù)成員屬性名去字典中查找對應(yīng)的value
        id value = dic[key];
        
        // 獲取成員屬性類型
        NSString * propertyType = [NSString stringWithUTF8String:ivar_getTypeEncoding(ivar)];
        
        // 二級轉(zhuǎn)換:如果字典中還有字典,也需要把對應(yīng)的字典轉(zhuǎn)換成模型
        // 判斷下value是否是字典
        if ([value isKindOfClass:[NSDictionary class]]) {
            // 字典轉(zhuǎn)模型
            // 1.獲取模型的類對象,調(diào)用modelWithDict
            // 2.模型的類名已知,就是成員屬性的類型
            
            // 獲取成員屬性類型
            // 生成的是這種@"@\"User\"" 類型 -》 @"User"  在OC字符串中 \" -> ",\是轉(zhuǎn)義的意思,不占用字符
            // 裁剪類型字符串
            NSRange range = [propertyType rangeOfString:@"\""];
            
            propertyType = [propertyType substringFromIndex:range.location + range.length];
            
            range = [propertyType rangeOfString:@"\""];
            
            // 裁剪到哪個角標(biāo),不包括當(dāng)前角標(biāo)
            propertyType = [propertyType substringToIndex:range.location];
            
            // 根據(jù)字符串類名生成類對象
            Class modelClass = NSClassFromString(propertyType);
            
            // 有對應(yīng)的模型才需要轉(zhuǎn)
            if (modelClass) {
               // 把字典轉(zhuǎn)模型
                value = [modelClass modelWithDic:value];
            }
            
            // 三級轉(zhuǎn)換:NSArray中也是字典,把數(shù)組中的字典轉(zhuǎn)換成模型.
            // 判斷值是否是數(shù)組
        }else if ([value isKindOfClass:[NSArray class]]){
            
            // 判斷對應(yīng)類有沒有實(shí)現(xiàn)字典數(shù)組轉(zhuǎn)模型數(shù)組的協(xié)議
            if ([self respondsToSelector:@selector(arrayContainModelClass)]) {
                
                // 轉(zhuǎn)換成id類型,就能調(diào)用任何對象的方法
                id idSelf = self;
                
                // 獲取數(shù)組中字典對應(yīng)的模型
                NSString *type = [idSelf arrayContainModelClass][key];
                
                // 生成模型
                Class classModel = NSClassFromString(type);
                
                NSMutableArray *muArray = [NSMutableArray array];
                
                // 遍歷字典數(shù)組,生成模型數(shù)組
                for (NSDictionary * dic in value) {
                    
                    // 字典轉(zhuǎn)模型
                    id model = [classModel modelWithDic:dic];
                    [muArray addObject:model];
                }
                
                // 把模型數(shù)組賦值給value
                value = muArray;
                
            }
        }
        
        // 有值,才需要給模型的屬性賦值
        // 利用KVC給模型中的屬性賦值
        if (value) {
            [objc setValue:value forKey:key];
        }
        
    }
    
    return objc;
}

@end

完整字典轉(zhuǎn)模型代碼請點(diǎn)這里這里

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,963評論 6 542
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 99,348評論 3 429
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,083評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,706評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 72,442評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,802評論 1 328
  • 那天,我揣著相機(jī)與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,795評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,983評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 49,542評論 1 335
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 41,287評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點(diǎn)故事閱讀 43,486評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,030評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 44,710評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,116評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,412評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,224評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 48,462評論 2 378

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

  • 轉(zhuǎn)至元數(shù)據(jù)結(jié)尾創(chuàng)建: 董瀟偉,最新修改于: 十二月 23, 2016 轉(zhuǎn)至元數(shù)據(jù)起始第一章:isa和Class一....
    40c0490e5268閱讀 1,755評論 0 9
  • 本文詳細(xì)整理了 Cocoa 的 Runtime 系統(tǒng)的知識,它使得 Objective-C 如虎添翼,具備了靈活的...
    lylaut閱讀 819評論 0 4
  • 文中的實(shí)驗代碼我放在了這個項目中。 以下內(nèi)容是我通過整理[這篇博客] (http://yulingtianxia....
    茗涙閱讀 938評論 0 6
  • 轉(zhuǎn)載:http://yulingtianxia.com/blog/2014/11/05/objective-c-r...
    F麥子閱讀 758評論 0 2
  • 游居在高原的丘陵溝壑之城已經(jīng)一年四個月十八天,仔細(xì)算來還需再加十五天寶塔山下的生活,延安這個地方是英雄的城孕...
    流曦溪閱讀 312評論 0 0