美文网首页iOS超棒文集i like code
NSMethodSignature和NSInvocation的一

NSMethodSignature和NSInvocation的一

作者: 城市之光 | 来源:发表于2016-01-08 13:54 被阅读6726次

    NSMethodSignature顾名思义应该就是“方法签名”,类似于C++中的编译器时的函数签名。苹果官方定义该类为对方法的参数、返回类似进行封装,协同NSInvocation实现消息转发。通过消息转发实现类似C++中的多重继承。

    iOS中的SEL,它的作用和C、C++中的函数指针很相似,通过performSelector:withObject:函数可以直接调用这个消息。但是perform相关的这些函数,有一个局限性,其参数数量不能超过2个,否则要做很麻烦的处理,与之相对,NSInvocation也是一种消息调用的方法,并且它的参数没有限制。这两种直接调用对象消息的方法,在IOS4.0之后,大多被block结构所取代,只有在很老的兼容性系统中才会使用。
    简单用法如下:
    - (void)test {
    int a = 1;
    int b = 2;
    int c = 3;
    SEL myMethod = @selector(myLog:param:parm:);
    SEL myMethod2 = @selector(myLog);
    // 创建一个函数签名,这个签名可以是任意的,但需要注意,签名函数的参数数量要和调用的一致。
    NSMethodSignature *sig = [[self class] instanceMethodSignatureForSelector:myMethod];
    // 通过签名初始化
    NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:sig];
    // 2.FirstViewController *view = self;
    // 2.[invocation setArgument:&view atIndex:0];
    // 2.[invocation setArgument:&myMethod2 atIndex:1];
    // 设置target
    // 1.[invocation setTarget:self];
    // 设置selector
    [invocation setSelector:myMethod];
    // 注意:1、这里设置参数的Index 需要从2开始,因为前两个被selector和target占用。
    [invocation setArgument:&a atIndex:2];
    [invocation setArgument:&b atIndex:3];
    [invocation setArgument:&c atIndex:4];
    // [invocation retainArguments];
    // 我们将c的值设置为返回值
    [invocation setReturnValue:&c];
    int d;
    // 取这个返回值
    [invocation getReturnValue:&d];
    NSLog(@"d:%d", d);

       NSUInteger argCount = [sig numberOfArguments];
       NSLog(@"argCount:%ld", argCount);
    
       for (NSInteger i=0; i<argCount; i++) {
           NSLog(@"%s", [sig getArgumentTypeAtIndex:i]);
       }
       NSLog(@"returnType:%s ,returnLen:%ld", [sig methodReturnType], [sig methodReturnLength]);
       NSLog(@"signature:%@" , sig);
    
       // 消息调用
       2.[invocation invokeWithTarget:self];
    }
    
    注意:代码中用1.标识的为第一种用法,通过setTarget和setSelector来设置NSInvocation的参数,而用2.标识的是另一种用法,通过setArgument atIndex:来设置参数。看个人的喜好。。。
    
    - (int)myLog:(int)a param:(int)b parm:(int)c
    {
         NSLog(@"MyLog:%d,%d,%d", a, b, c);
         return a+b+c;
    }
    
    - (void)myLog
    {
         NSLog(@"你好,South China University of Technology");
    }
    

    消息转发中的使用:
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)selector
    {
    NSMethodSignature *signature = [super methodSignatureForSelector:selector];
    if (!signature) {
    signature = [target methodSignatureForSelector:selector];
    }
    return signature;
    }

    + (BOOL)resolveInstanceMethod:(SEL)sel
    {
          if (sel == @selector(foo)) {
          class_addMethod([self class], sel, (IMP)dynamicMethodIMP, "V@:");
          return YES;
          }
          return [super resolveInstanceMethod:sel];
    }
    
    - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
    {
           NSMethodSignature *signature = [super methodSignatureForSelector:aSelector];
           if (!signature) {
                for (id target in self.allDelegates) {
                     if ((signature = [target methodSignatureForSelector:aSelector])) {
                          break;
                     }
                }
           }
           return signature;
    }
    
    - (void)forwardInvocation:(NSInvocation *)anInvocation
    {
          for (id target in self.allDelegates) {
               if ([target respondsToSelector:anInvocation.selector]) {
                    [anInvocation invokeWithTarget:target];
               }
          }
    }
    
    - (BOOL)respondsToSelector:(SEL)aSelector
    {
          if ([super respondsToSelector:aSelector]) {
              return YES;
          }
    
          for (id target in self.allDelegates) {
               if ([target respondsToSelector:aSelector]) {
                    return YES;
               }
          }
          return NO;
    }
    

    备注: allDelegate是存储delegate的数组结构。

    相关文章

      网友评论

      • 韦德460:哥们,你这个消息转发的demo还有吗

      本文标题:NSMethodSignature和NSInvocation的一

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