我不明白为什么我们必须在setSelector
对象上调用NSInvocation
方法,因为该信息已经通过invocationWithMethodSignature
传递。
要创建NSInvocation
对象,我们会执行以下操作:
SEL someSelector;
NSMethodSignature *signature;
NSInvocation *invocation;
someSelector = @selector(sayHelloWithString:);
//Here we use the selector to create the signature
signature = [SomeObject instanceMethodSignatureForSelector:someSelector];
invocation = [NSInvocation invocationWithMethodSignature:signature];
//Here, we again set the same selector
[invocation setSelector:someSelector];
[invocation setTarget:someObjectInstance];
[invocation setArgument:@"Loving C" atIndex:2];
请注意,我们已将选择器传递给[SomeObject instanceMethodSignatureForSelector: someSelector];
并再次传递给[invocation setSelector:someSelector];
。
我有什么遗失的吗?
答案 0 :(得分:8)
签名不是选择器。选择器是消息的名称。签名定义参数和返回值。您可以拥有许多具有相同签名的选择器,反之亦然。如果您查看NSMethodSignature
,您会注意到没有-selector
方法;签名不随身携带特定的选择器。
考虑以下
- (void)setLocation:(CGFloat)aLocation;
- (void)setLocation:(MyLocation*)aLocation;
他们有相同的选择器@selector(setLocation:)
,但签名不同。
- (void)setX:(CGFloat)x;
- (void)setY:(CGFloat)y;
它们具有相同的签名,但选择器不同。
来自ObjC编程语言的Selectors可能是理解这一点的有用参考。
答案 1 :(得分:3)
方法签名仅定义返回类型以及参数的数量和类型。它不包含有关选择器名称的任何内容。例如,尽管有不同的选择器,所有这些方法都具有相同的签名:
-(void) foo:(NSString*)fooString;
-(void) bar:(NSString*)barString;
-(void) baz:(NSString*)bazString;
答案 2 :(得分:0)
这是一个侧面答案,但您可以执行以下操作这一事实有助于我更好地理解方法签名和选择器之间的分离。
此代码位于视图控制器
中NSMethodSignature *sig = nil;
sig = [[self class] instanceMethodSignatureForSelector:@selector(viewDidAppear:)];
NSInvocation *myInvocation = nil;
myInvocation = [NSInvocation invocationWithMethodSignature:sig];
[myInvocation setTarget:_somePopoverController];
[myInvocation setSelector:@selector(dismissPopoverAnimated:)];
BOOL animate = YES;
[myInvocation setArgument:&animate atIndex:2];
[myInvocation invoke];
由于UIViewController的viewDidAppear: 和UIPopoverController的dismissPopoverAnimated:都采用BOOL参数并返回void,您可以使用一个选择器创建方法签名,但将调用发送给另一个。