當(dāng)我們有方法名和參數(shù)列表拌消,想要動態(tài)地給對象發(fā)送消息,可用通過反射函數(shù)機(jī)制來實(shí)現(xiàn)割以,有兩種常用的做法:
一启具、performSelector
常用的方法有這三個(gè)本讥,其中aSelector可以通過 NSSelectorFromString 方法拿到
但是 performSelector 的缺點(diǎn)是最多只支持傳遞兩個(gè)參數(shù)
二、NSInvocation
// 測試反射函數(shù)
- (void)printWithString:(NSString *)string withNum:(NSNumber *)number withArray:(NSArray *)array {
NSLog(@"%@, %@, %@", string, number, array[1]);
}
- (void)test {
NSString *str = @"哈哈哈";
NSNumber *num = @20;
NSArray *arr = @[@"ABC", @"DEF"]; // [self printWithString:str withNum:num withArray:arr];
SEL sel = NSSelectorFromString(@"printWithString:withNum:withArray:");
NSArray *objs = [NSArray arrayWithObjects:str, num, arr, nil];
[self performSelector:sel withObjects:objs];
}
- (id)performSelector:(SEL)selector withObjects:(NSArray *)objects { // 方法簽名(方法的描述)
NSMethodSignature *signature = [[self class] instanceMethodSignatureForSelector:selector];
if (signature == nil) { //可以拋出異常也可以不操作鲁冯。
}
// NSInvocation : 利用一個(gè)NSInvocation對象包裝一次方法調(diào)用(方法調(diào)用者拷沸、方法名、方法參數(shù)薯演、方法返回值)
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:signature];
invocation.target = self;
invocation.selector = selector; // 設(shè)置參數(shù)
NSInteger paramsCount = signature.numberOfArguments - 2; // 除self撞芍、_cmd以外的參數(shù)個(gè)數(shù)
paramsCount = MIN(paramsCount, objects.count);
for (NSInteger i = 0; i < paramsCount; i++) {
id object = objects[i];
if ([object isKindOfClass:[NSNull class]]) continue;
[invocation setArgument:&object atIndex:i + 2];
} // 調(diào)用方法
[invocation invoke]; // 獲取返回值
id returnValue = nil; if (signature.methodReturnLength) { // 有返回值類型,才去獲得返回值
[invocation getReturnValue:&returnValue];
} return returnValue;
}
三跨扮、objc_msgSend
objc_msgSend的寫法要復(fù)雜一點(diǎn)序无,具體可以參看這篇文章,講的很清楚
但是有個(gè)缺點(diǎn)是衡创,需要指定好傳遞參數(shù)的類型帝嗡,是不是可以直接都用id呢?
經(jīng)測試id可用
// objc_msgSend
SEL sel = NSSelectorFromString(@"printWithString:withNum:withArray:");
((void (*) (id, SEL, NSString *, NSNumber *, NSArray *)) objc_msgSend) (self, sel, str, num, arr);