iOS runtime機制和使用
阿新 • • 發佈:2018-12-11
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 來源:簡書