Какова цель метода setSelector в классе NSInvocation?

Я не понимаю, почему мы должны вызывать метод 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];.

Есть ли что-то, что мне не хватает?

Ответ 1

Подпись не является селектором. Селектор - это имя сообщения. Подпись определяет параметры и возвращаемое значение. У вас может быть множество селекторов с одной и той же подписью и наоборот. Если вы посмотрите NSMethodSignature, вы заметите, что нет метода -selector; подписи не переносятся вокруг определенного селектора.

Рассмотрим следующее

- (void)setLocation:(CGFloat)aLocation;
- (void)setLocation:(MyLocation*)aLocation;

Они имеют один и тот же селектор @selector(setLocation:), но разные подписи.

- (void)setX:(CGFloat)x;
- (void)setY:(CGFloat)y;

Они имеют одну и ту же подпись, но разные селекторы.

Selectors с языка программирования ObjC может быть полезной ссылкой для понимания этого.

Ответ 2

Подписи метода определяют только тип возвращаемого значения, а также количество и тип аргументов. Он не содержит ничего о имени селектора. Например, все эти методы имеют одну и ту же подпись, несмотря на наличие разных селекторов:

-(void) foo:(NSString*)fooString;
-(void) bar:(NSString*)barString;
-(void) baz:(NSString*)bazString;

Ответ 3

Это своего рода побочный ответ, но тот факт, что вы можете сделать следующее, помог мне лучше понять разделение между сигнатурами методов и селекторами.

Этот код находится в контроллере просмотра

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 rejectPopoverAnimated: оба принимают аргумент BOOL и возвращают void, вы можете создать подпись метода с помощью одного селектора, но отправить вызов другому.