一枢步、關于runtime簡介
runtime? 顧名思義,就是運行時的意思,是一套底層的 C 語言 API庇谆,其為 iOS 內(nèi)部的核心之一忌锯,我們平時編寫的 OC 代碼幔翰,底層都是基于它來實現(xiàn)的。我們需要了解的是 Objective-C 是一門動態(tài)語言西壮,它會將一些工作放在代碼運行時才處理而并非編譯時遗增。
OC就是運行時機制,也就是在運行時候的一些機制款青,其中最主要的是消息機制做修。
在C語言中,函數(shù)的調(diào)用在編譯的時候會決定調(diào)用哪個函數(shù)抡草。而對于OC的函數(shù)饰及,屬于動態(tài)調(diào)用過程,在編譯的時候并不能決定真正調(diào)用哪個函數(shù)康震,只有在真正運行的時候才會根據(jù)函數(shù)的名稱找到對應的函數(shù)來調(diào)用燎含。
其實事實上在編譯階段,OC可以調(diào)用任何函數(shù)腿短,即使這個函數(shù)并未實現(xiàn)屏箍,只要聲明過就不會報錯
而在c語言中 ,在編譯階段,C語言調(diào)用未實現(xiàn)的函數(shù)就會報錯.
二橘忱、關于runtime作用
由于運行時的應用比較廣泛,我們暫時選擇幾種比較常見的容易理解的來介紹
1.發(fā)送消息
? 方法調(diào)用的本質(zhì)赴魁,就是讓對象發(fā)送消息。使用消息機制前提钝诚,必須導入#import<objc/message.h>
?消息機制原理:對象根據(jù)方法編號SEL去映射表查找對應的方法實現(xiàn)
Student* stu=[[Student alloc]init];
[stu study];
// 本質(zhì):讓對象發(fā)送消息
objc_msgSend(stu, @selector(study));
2.動態(tài)添加方法
如果聲明了某個方法 但是沒有實現(xiàn) ,編譯成功,到那時運行會崩掉
2017-08-30 12:01:00.453 runtimeTest[3228:71218] -[Student study]: unrecognized selector sent to instance 0x60800001ece0
2017-08-30 12:01:00.460 runtimeTest[3228:71218] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[Student study]: unrecognized selector sent to instance 0x60800001ece0'
Student* stu=[[Student alloc]init];
[stu study];
//如果實現(xiàn)了study 打印如下:? 2017-08-30 13:30:01.970 runtimeTest[4004:94732] 正在學習
// 動態(tài)添加方法就不會報錯? 可以通過performSelector調(diào)用颖御,但是會報錯。
[stu performSelector:@selector(study)];
在student.m 中
@implementation Student
//-(void)study
//{
//? ? NSLog(@"正在學習");
//}
//動態(tài)添加
void study(id self ,SEL sel){
//打印出方法名
NSLog(@"%@ %@",self,NSStringFromSelector(sel));
}
// 當一個對象調(diào)用未實現(xiàn)的方法凝颇,會調(diào)用這個方法處理,并且會把對應的方法列表傳過來.
// 剛好可以用來判斷潘拱,未實現(xiàn)的方法是不是我們想要動態(tài)添加的方法
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
if (sel == @selector(study)) {
// 動態(tài)添加eat方法
// 第一個參數(shù):給哪個類添加方法
// 第二個參數(shù):添加方法的方法編號
// 第三個參數(shù):添加方法的函數(shù)實現(xiàn)(函數(shù)地址)
// 第四個參數(shù):函數(shù)的類型,(返回值+參數(shù)類型) v:void @:對象->self :表示SEL->_cmd
class_addMethod(self, @selector(study), study, "v@:");
}
return [super resolveInstanceMethod:sel];
}
@end
再次運行打印結(jié)果如下:
2017-08-30 13:33:38.066 runtimeTest[4093:96918]study
所謂動態(tài)添加就是 + (BOOL)resolveInstanceMethod:(SEL)sel 方法中判斷方法是不是我們需要的,如果是就class_addMethod
3 ?給分類添加屬性
也就是給一個類聲明屬性拧略,其實本質(zhì)就是給這個類添加關聯(lián)
我創(chuàng)建了一個User 類 ,什么都沒添加,并且給創(chuàng)建了一個分類User (Name)
#import "User.h"
@interface User (Name)
@property(nonatomic,strong)NSString* name;
@end
一開始并不實現(xiàn)屬性的方法直接調(diào)用
結(jié)果如下:
2017-08-30 13:45:42.225 runtimeTest[4267:102049] -[User setName:]: unrecognized selector sent to instance 0x608000007f30
2017-08-30 13:45:42.229 runtimeTest[4267:102049] *** Terminating app due to uncaught exception 'NSInvalidArgumentException', reason: '-[User setName:]: unrecognized selector sent to instance 0x608000007f30'
崩潰是在意料之中的
我們再在.m文件做如下處理
#import "User+Name.h"
// 定義關聯(lián)的key
static? char *keyName = "name";
@implementation User (Name)
-(NSString*)name
{
// return objc_getAssociatedObject(<#id object#>, <#const void *key#>)
//第一個參數(shù) 關聯(lián)的object
//第二個參數(shù) 關聯(lián)的key
return objc_getAssociatedObject(self, keyName);
}
-(void)setName:(NSString *)name
{
//objc_setAssociatedObject(<#id object#>, <#const void *key#>, <#id value#>, <#objc_AssociationPolicy policy#>)
// 第一個參數(shù):給哪個對象添加關聯(lián)
// 第二個參數(shù):關聯(lián)的key泽铛,通過這個key獲取
// 第三個參數(shù):關聯(lián)的value
// 第四個參數(shù):關聯(lián)的策略
objc_setAssociatedObject(self, keyName, name, OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}
@end
然后新運行
2017-08-30 13:56:27.299 runtimeTest[4477:107135] ---蚊子---蚊子不吸血
? 4 ? 交換方法
聲明一個分類 NSString+More
#import "NSString+More.h"
#import <objc/runtime.h>
@implementation NSString (More)
//重新添加的一個方法用于交換
+ (void)makestringWithString:(NSString *)string
{
NSLog(@"新方法----%@",string);
NSLog(@"新方法----%@",self);
}
//重寫load方法
+ (void)load
{
//獲取原來containsString的地址
Method containsString=class_getClassMethod(self, @selector(stringWithString:));
//獲取原來addcontainsString的地址
Method addcontainsString=class_getClassMethod(self, @selector(makestringWithString:));
// 交換方法地址,相當于交換實現(xiàn)方式
method_exchangeImplementations(containsString, addcontainsString);
}
@end
調(diào)用方法
NSString* str1=@"23333";
[NSString stringWithString:str1];
結(jié)果如下 ?可以看到執(zhí)行了我們在.m 中實現(xiàn)的方法
2017-08-30 14:23:07.596 runtimeTest[4929:120296] 新方法----23333
2017-08-30 14:23:07.597 runtimeTest[4929:120296] 新方法----NSString
這些基本上足夠淺淺 的入門了,暫時就先介紹到這里 ,蚊子君以后空閑有了新的理解會持續(xù)更新