如何通过消息转发"吞下""无法识别的选择器错误"?

Jie*_* Wu 5 objective-c message-forwarding

我只是在研究Objective-C的"消息转发".我编写了一个测试程序来验证我是否可以在运行时"吞下"一个无法识别的选择器.所以我这样做了:

- (void) forwardInvocation: (NSInvocation *) anInvocation {
    if ([anInvocation selector] == @selector(testMessage)){
       NSLog(@"Unknow message");
    }
    return;
}
Run Code Online (Sandbox Code Playgroud)

但它仍然在运行时抛出"无法识别的选择器"错误.在搜索了解决方案后,我知道我需要覆盖方法"methodSignatureForSelector:",所以我编写了另一个名为"Proxy"的代理类,并使用以下方法:

(NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
    if ([Proxy instancesRespondToSelector: selector]) {
        return [Proxy instanceMethodSignatureForSelector: selector];
    }   
    return [super methodSignatureForSelector:selector];
}
Run Code Online (Sandbox Code Playgroud)

但实际上,我不想实现这样的另一个代理类来完成这个方法.我想做的就是忽略这个未知的选择器.但是,如果我只是键入它,它不起作用:

(NSMethodSignature *)methodSignatureForSelector:(SEL)selector {   
    return [super methodSignatureForSelector:selector];
}
Run Code Online (Sandbox Code Playgroud)

所以,我想知道有什么办法可以简单地"吞下"这个错误吗?(不使用异常处理程序,我想采用"转发"方式).谢谢!

Dar*_*ust 3

您可能需要生成一个签名对象,这有点棘手:

- (NSMethodSignature *)methodSignatureForSelector:(SEL)selector {
    NSMethodSignature *sig = [super methodSignatureForSelector:selector];
    if (!sig) {
        if (selector == @selector(testMessage)) {
            // Create a signature for our fake message.
            sig = [NSMethodSignature signatureWithObjCTypes:"@:"];
        }
    }
    return sig;
}

- (void)forwardInvocation:(NSInvocation *)anInvocation
{
   // Do whatever you want with the invocation.
}
Run Code Online (Sandbox Code Playgroud)

最大的问题似乎是 的争论signatureWithObjCTypes:。请参阅Apple 的类型编码文档。每个方法至少有两个参数:id self(type @) 和SEL _cmd(type :)。

另一种方法是使用一个虚拟方法,- (void)dummy { }然后使用它sig = [super methodSignatureForSelector:@selector(dummy)];来获取伪造方法的签名。

  • 我发现[这个](http://www.cocoabuilder.com/archive/cocoa/289799-nsproxy-nsinvocau-question.html):*[NSMethodSignaturesignatureWithObjTypes:]的文档是错误的。类型数组中的第一个元素是返回类型,而不是 id 或 Class 类型,这实际上给我留下了一个 MSMethodSignature 实例,其中不包含选择器参数的参数类型。*这可能意味着对于方法 `-(void) foo` 您需要在 `signatureWithObjCTypes:` 的参数中使用类型 `"v@:"`。 (3认同)