我无法让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'
答案 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。