1. 程式人生 > >iOS runtime機制和使用

iOS runtime機制和使用

runtime簡稱執行時。OC是執行時機制,也就是在執行時才做一些處理。例如:C語言在編譯的時候就知道要呼叫哪個方法函式,而OC在編譯的時候並不知道要呼叫哪個方法函式,只有在執行的時候才知道呼叫的方法函式名稱,來找到對應的方法函式進行呼叫。

匯入

想要使用runtime,就要先匯入runtime庫 一般匯入message.h,因為message.h包含了objc.h和runtime.h

#import “<objc/message.h>”

runtime作用

一:傳送訊息

OC方法呼叫原理

方法呼叫的本質就是放物件傳送訊息

/* new 會呼叫 init方法 */
People *man = [People new];
People *man = [[People alloc] init];

//屬性方法呼叫的方式
[man eat];
//類方法呼叫方式
[People eat];
[[People class] eat];

//還有一種不常用的呼叫方式
[物件/類 performSelector:@selector(eat)];

//底層實現
objc_msgSend(物件/屬性, @selector(eat));

最終程式碼檢視方法:clang -rewrite-objc main.m 訊息傳送底層實現 Build Setting設定msg為NO Xcode5之後使用runtime機制 OC 與 C 的對應方法 [People class] == objc_getClass("People") @selector() == sel_registerName()

二:交換方法

當系統自帶的方法功能不夠,可以給系統自帶的方法擴充套件一些功能,並保持原有的功能

例如我想知道當前的URL是否為空如果每次都判斷一下的話會很麻煩,如果我建立擴充套件來寫,又不知道內部是如何實現的.

一:使用繼承來實現

//.h檔案內容
#import <Foundation/Foundation.h>

@interface CFURL : NSURL

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

@end

-----------------------------------------------
//.m檔案內容
#import "CFURL.h"

@implementation CFURL

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

@end

二:使用runtime交換方法

//.h檔案內容
#import <Foundation/Foundation.h>

@interface NSURL (url)

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

@end

-----------------------------------------------
//.m檔案內容
#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];//注意這裡不能再呼叫系統的方法
    if (!url) {
        NSLog(@"url為空");
    }
    return url;
}

@end

三:動態新增方法

//.m檔案
#import "People.h"

#import <objc/runtime.h>

@implementation People

//當類呼叫一個沒有實現的類方法就會到這裡!!
+(BOOL)resolveClassMethod:(SEL)sel{
    NSLog(@"%@",NSStringFromSelector(sel));
    return [super resolveClassMethod:sel];
}

//當類呼叫一個沒有實現的物件方法就會到這裡!!
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
    if (sel == @selector(eat)) {
        // 動態新增eat方法
       /*
         第一個引數:給哪個類新增方法
         第二個引數:新增方法的方法編號
         第三個引數:新增方法的函式實現(函式地址)
         第四個引數:函式的型別,(返回值+引數型別) v:void @:物件->self :表示SEL->_cmd
       */
        class_addMethod(self, @selector(eat), eat, "[email protected]:");
    }
    return [super resolveInstanceMethod:sel];
}

// 預設方法都有兩個隱式引數,
void eat(id self,SEL sel)
{
    NSLog(@"%@ %@",self,NSStringFromSelector(sel));
}

@end

四: 給分類新增屬性

//.h檔案
#import "NSObject.h"

@interface NSObject (Property)

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

@end

-----------------------------------------------
//.m檔案
// 定義關聯的key
static const char *key = "name";

@implementation NSObject (Property)

- (NSString *)name
{
    // 根據關聯的key,獲取關聯的值。
    return objc_getAssociatedObject(self, key);
}

- (void)setName:(NSString *)name
{
    /*
     第一個引數:給哪個物件新增關聯
     第二個引數:關聯的key,通過這個key獲取
     第三個引數:關聯的value
     第四個引數:關聯的策略
   */
    objc_setAssociatedObject(self, key, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

@end

作者:遛遛食 連結:https://www.jianshu.com/p/4bae6de3b11c 來源:簡書