RuntimeInvoker是一个Objective-C反射调用库的封装。
https://github.com/cyanzhong/RuntimeInvoker
以下是一些简单的使用举例:
-(void)test{
NSString *s = [self invoke:@"fun1"];
NSLog(@"---------%@",s);
}
-(NSString *)fun1{
return @"fun1";
}
-(void)test{
[self invoke:@"fun2:" args:@(YES),nil];
}
-(void)fun2:(BOOL)bo{
if (bo) {
NSLog(@"fun2 yes");
}else{
NSLog(@"fun2 no");
}
}
-(void)test{
NSString *s = [[self class] invoke:@"fun3:" args:@"abc",nil];
NSLog(@"---------%@",s);
}
+(NSString *)fun3:(NSString *)str{
return [NSString stringWithFormat:@"fun3 %@",str];
}
-(void)test{
NSString *s = [@"AppDelegate" invokeClassMethod:@"fun4:" args:@"abc",nil];
NSLog(@"---------%@",s);
}
+(NSString *)fun4:(NSString *)str{
return [NSString stringWithFormat:@"fun4 %@",str];
}
如果类是由字符串过去的,需要用invokeClassMethod替换invoke。
遇到这样一个问题,方法如下:
-(void)callFun6:(NSString *)str1 index:(int)index f:(CGFloat)f point:(CGPoint)p;
我在拿到类对象的情况下,需要调用这个函数,该怎么做?
见下面代码:
TartetObject *target = [TartetObject new];
id obj = target;
NSString *str1 = @"2";
NSInteger index = 3;
CGFloat f = 3.1415926;
NSString *selStr = @"callFun6:index:f:point:";
// SEL sel = NSSelectorFromString(selStr);
CGPoint point = CGPointMake(32.121, 195.321);
[obj invoke:selStr args:str1,[NSNumber numberWithInt:index],[NSNumber numberWithFloat:f],[NSValue valueWithCGPoint:point],nil];
执行,会发现CGpoint没有传值成功。阅读RuntimeInvoker代码可以发现作者在invocationWithArguments函数中少写了一部分代码。
后面补上point的类型
case RIMethodArgumentTypeCGPoint: {
CGPoint p = [argument CGPointValue];
[invocation setArgument:&p atIndex:index];
} break;
再跑一遍,传参数成功了!
如果参数带有block该怎么处理呢?
函数原型如下:
-(void)callFun6:(NSString *)str1 index:(int)index f:(CGFloat)f point:(CGPoint)p callBack:(void(^)(NSString *))block{
NSLog(@"callFun6 = %@%d%.2f;x=%.2fy=%.2f",str1,index,f,p.x,p.y);
block(@"abcdefg");
}
调用他:
TartetObject *target = [TartetObject new];
id obj = target;
NSString *str1 = @"2";
NSInteger index = 3;
CGFloat f = 3.1415926;
NSString *selStr = @"callFun6:index:f:point:callBack:";
CGPoint point = CGPointMake(32.121, 195.321);
void (^callblock)(NSString*) = ^(NSString * str){
NSLog(@"------------%@",str);
[target callFun1];
};
[obj invoke:selStr args:str1,[NSNumber numberWithInt:index],[NSNumber numberWithFloat:f],[NSValue valueWithCGPoint:point],callblock,nil];
发现出错了,block传过来是空的。
看下RuntimeInvoker代码,是的确没有处理参数是block的情况。我们手动改一下,代码如下:
在判断为未知类型的情况下,看下这个类是否是block。如果是block,就直接把参数传递过去。当然,在未知情况下直接把参数传递过去也是可以的。
好了,RuntimeInvoker基本上可以满足我们日常使用需求了。