据我目前所学:在 Objective-C 中,您可以向任何对象发送任何消息。如果对象确实实现了正确的方法,它将被执行,否则什么也不会发生。这是因为在发送消息之前,Objective-C 会执行respondsToSelector。
我希望到目前为止我是对的。
我做了一个小程序来测试每次移动 slider 时都会调用一个 Action 。同样为了测试,我将发件人设置为 NSButton,但实际上它是一个 NSSlider。现在我询问对象是否会响应 setAlternateTitle。虽然 NSButton 可以,但 NSSlider 不会。如果我运行代码并自己执行 respondsToSelector,它会告诉我该对象不会响应该选择器。如果我测试诸如 intValue 之类的其他东西,它就会响应。所以到目前为止,我的代码还不错。
- (IBAction)sliderDidMove:(id)sender
{
NSButton *slider = sender;
BOOL responds =
[slider respondsToSelector:@selector(setAlternateTitle)];
if(responds == YES)
{
NSLog(@"YES");
}
else
{
NSLog(@"NO");
}
[slider setAlternateTitle:@"Hello World"];
}
但是当我实际发送 setAlternateTitle 消息时,程序会崩溃,我不确定原因。它不应该在发送消息之前执行 respondsToSelector 吗?
最佳答案
首先,方法的名称(它的选择器)包括所有子部分和冒号字符,如 mvds 所说。
其次,-respondsToSelector:
方法不被运行时调用,它通常由用户调用(您自己或想知道委托(delegate)是否响应的 API,例如,协议(protocol)的可选方法)。
当你向对象发送消息时,运行时会在对象的类中查找方法的实现(通过对象的isa指针)。它等同于发送 -respondsToSelector:
尽管消息本身没有被发送。如果在类或其父类(super class)中找到该方法的实现,则会使用您传入的所有参数调用它。
如果不是,则运行时会为消息提供第二次执行机会。它首先将消息 + (BOOL)resolveInstanceMethod:(SEL)name
发送到对象的类:此方法允许您在运行时将方法添加到类:如果此消息返回 YES ,这意味着它可以重新发送消息。
如果没有,它给消息第三次执行机会,它发送 - (id)forwardingTargetForSelector:(SEL)aSelector
和选择器,这个方法可以返回另一个对象,可能能够代表实际接收者响应选择器,如果返回的对象可以响应,则执行该方法并返回值,就好像它是由原始消息返回的一样。 (注意:从 OS X 10.6 或 iOS 4 开始可用。)
如果返回的对象是 nil 或 self(避免无限循环),运行时给消息第四次机会来执行该方法……它发送消息 - (NSMethodSignature *)methodSignatureForSelector:(SEL)aSelector
获取方法签名以构建调用。如果提供了一个调用,则通过消息 - (void)forwardInvocation:(NSInvocation *)anInvocation
发送调用。在这个方法中,您可以解析调用并构建其他消息以您想要的任何方式发送到其他目标,然后您可以设置调用的返回值……该值将作为原始消息的返回值。
最后,如果对象没有返回方法签名,那么运行时发送消息- (void)doesNotRecognizeSelector:(SEL)aSelector
给你的对象,在NSObject类中实现这个方法抛出异常。
关于Objective-C 响应选择器,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/4574465/