美文网首页
iOS关于RuntimeInvoker

iOS关于RuntimeInvoker

作者: 学无止境吧 | 来源:发表于2017-02-04 12:56 被阅读56次

    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基本上可以满足我们日常使用需求了。

    相关文章

      网友评论

          本文标题:iOS关于RuntimeInvoker

          本文链接:https://www.haomeiwen.com/subject/klbjittx.html