forwardInvocation没有被调用?

时间:2011-01-06 23:05:29

标签: iphone objective-c selector nsinvocation

我无法让forwardInvocation工作。出于某种原因,Objective-C运行时完全忽略了我的forwardInvocation:方法并抛出了一个无法识别的选择器异常。

我的测试代码如下:

@interface InvocationTest : NSObject
{
}

+ (void) runTest;

@end


@interface FullClass: NSObject
{
    int value;
}
@property(readwrite,assign) int value;

@end

@implementation FullClass

@synthesize value;

@end


@interface SparseClass: NSObject
{
}

@end

@implementation SparseClass

- (void)forwardInvocation:(NSInvocation *)forwardedInvocation
{
    NSLog(@"ForawrdInvocation called");

    FullClass* proxy = [[[FullClass alloc] init] autorelease];
    proxy.value = 42;
    [forwardedInvocation invokeWithTarget:proxy];
}

@end


@implementation InvocationTest

+ (void) runTest
{
    SparseClass* sparse = [[[SparseClass alloc] init] autorelease];
    NSLog(@"Value = %d", [sparse value]);
}

@end

我正在处理来自以下资源的信息:

http://developer.apple.com/library/mac/#documentation/Cocoa/Conceptual/ObjCRuntimeGuide/Articles/ocrtForwarding.html#//apple_ref/doc/uid/TP40008048-CH105 http://cocoawithlove.com/2008/03/construct-nsinvocation-for-any-message.html

据我所知,当我调用[稀疏值]时,运行时应该在SparseClass的实例上调用forwardInvocation:但是它被完全忽略了:

  

- [SparseClass value]:无法识别的选择器发送到实例0x4b1c4a0   ***由于未捕获的异常'NSInvalidArgumentException'终止应用程序,原因:' - [SparseClass值]:无法识别的选择器发送到实例0x4b1c4a0'

2 个答案:

答案 0 :(得分:38)

您还必须覆盖- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector才能使其正常运行。

我想

- (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector {
    return [FullClass instanceMethodSignatureForSelector:aSelector];
}

应该没问题。

答案 1 :(得分:25)

来自NSObject文档:

  

重要提示:要回复您的对象本身无法识别的方法,除了methodSignatureForSelector:之外,您还必须覆盖forwardInvocation:。转发消息的机制使用从methodSignatureForSelector:获取的信息来创建要转发的NSInvocation对象。您的重写方法必须为给定的选择器提供适当的方法签名,方法是预先制定一个或者通过询问另一个对象。

并且,来自runtime文档:

  

...如果一个对象转发它收到的任何远程消息,它应该有一个methodSignatureForSelector:版本,可以返回最终响应转发消息的方法的准确描述;例如,如果某个对象能够将消息转发给其代理,您可以按如下方式实现methodSignatureForSelector:

- (NSMethodSignature*)methodSignatureForSelector:(SEL)selector 
{ 
    NSMethodSignature* signature = [super methodSignatureForSelector:selector];
    if (!signature) {
       signature = [surrogate methodSignatureForSelector:selector];
    } 
    return signature;
}

注意:有关正确实施methodSignatureForSelector:的信息,请参阅Jilouc的answer