#import "ViewController.h"
#import "Person.h"
/*
1:
Runtime(動態(tài)添加方法):OC都是懶加載機制,只要一個方法實現(xiàn)了,就會馬上添加到方法列表中.
app:免費版,收費版
QQ,微博,直播等等應(yīng)用,都有會員機制
performSelector:去執(zhí)行某個方法冀墨。performSelector withObject :object為前面方法的參數(shù)
2:
美團(tuán)有個面試題?有沒有使用過performSelector,什么時候使用?動態(tài)添加方法的時候使用過?怎么動態(tài)添加方法?用runtime?為什么要動態(tài)添加方法?
*/
@interface ViewController ()
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// _cmd:當(dāng)前方法的方法編號
Person *p = [[Person alloc] init];
// 執(zhí)行某個方法
// [p performSelector:@selector(eat)];
[p performSelector:@selector(run:) withObject:@10];
}
@end
#import "Person.h"
#import <objc/message.h>
@implementation Person
// 沒有返回值,也沒有參數(shù)
// void,(id,SEL)
void aaa(id self, SEL _cmd, NSNumber *meter) {
NSLog(@"跑了%@", meter);
}
任何方法默認(rèn)都有兩個隱式參數(shù),self,_cmd(_cmd代表方法編號闸衫,打印結(jié)果為當(dāng)前執(zhí)行的方法名)
什么時候調(diào)用:只要一個對象調(diào)用了一個未實現(xiàn)的方法就會調(diào)用這個方法,進(jìn)行處理
作用:動態(tài)添加方法,處理未實現(xiàn)
+ (BOOL)resolveInstanceMethod:(SEL)sel
{
// [NSStringFromSelector(sel) isEqualToString:@"eat"];
if (sel == NSSelectorFromString(@"run:")) {
// eat
// class: 給哪個類添加方法
// SEL: 添加哪個方法
// IMP: 方法實現(xiàn) => 函數(shù) => 函數(shù)入口 => 函數(shù)名
// type: 方法類型:void用v來表示,id參數(shù)用@來表示诽嘉,SEL用:來表示
//aaa不會生成方法列表
class_addMethod(self, sel, (IMP)aaa, "v@:@");
return YES;
}
return [super resolveInstanceMethod:sel];
}
//- (void)test
//{
// // [NSStringFromSelector(sel) isEqualToString:@"eat"];
// if (sel == NSSelectorFromString(@"eat")) {
// // eat
// // class: 給哪個類添加方法
// // SEL: 添加哪個方法
// // IMP: 方法實現(xiàn) => 函數(shù) => 函數(shù)入口 => 函數(shù)名
// // type: 方法類型
// class_addMethod(self, sel, (IMP)aaa, "v@:");
//
// return YES;
// }
//
// return [super resolveInstanceMethod:sel];
//}
@end
3.動態(tài)添加方法
開發(fā)使用場景:如果一個類方法非常多蔚出,加載類到內(nèi)存的時候也比較耗費資源弟翘,需要給每個方法生成映射表,可以使用動態(tài)給某個類骄酗,添加方法解決稀余。
經(jīng)典面試題:有沒有使用performSelector,其實主要想問你有沒有動態(tài)添加過方法趋翻。
簡單使用
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
// Do any additional setup after loading the view, typically from a nib.
Person *p = [[Person alloc] init];
// 默認(rèn)person睛琳,沒有實現(xiàn)eat方法,可以通過performSelector調(diào)用踏烙,但是會報錯掸掏。
// 動態(tài)添加方法就不會報錯
[p performSelector:@selector(eat)];
}
@end
@implementation Person
// void(*)()
// 默認(rèn)方法都有兩個隱式參數(shù),
void eat(id self,SEL sel)
{
NSLog(@"%@ %@",self,NSStringFromSelector(sel));
}
當(dāng)一個對象調(diào)用未實現(xiàn)的方法宙帝,會調(diào)用這個方法處理,并且會把對應(yīng)的方法列表傳過來.
剛好可以用來判斷,未實現(xiàn)的方法是不是我們想要動態(tài)添加的方法
+ (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 YES;
}
return [super resolveInstanceMethod:sel];
}
@end