对于performSelector:
的结果,float/double/CGFloat类型转换有一个非常奇怪的行为。为什么这个工作?
BOOL property = (BOOL)[self.object performSelector:@selector(boolProperty)];
NSInteger property = (NSInteger) [self.object performSelector:@selector(integerProperty)];
这个不
CGFloat property = (CGFloat) [self.object performSelector:@selector(floatProperty)];
一开始我试着这样做:
CGFloat property = [[self.object performSelector:@selector(floatProperty)] floatValue];
但是我最终得到EXC_BAD_ACCESS运行时错误。我已经想出了一个hack来解决这个问题,但我想了解为什么它适用于整数和Bool,而不是浮点类型。
My hack:
@implementation NSObject (AddOn)
-(CGFloat)performFloatSelector:(SEL)aSelector
{
NSInvocation *invocation = [NSInvocation invocationWithMethodSignature:[[self class] instanceMethodSignatureForSelector:aSelector]];
[invocation setSelector:aSelector];
[invocation setTarget:self];
[invocation invoke];
CGFloat f = 0.0f;
[invocation getReturnValue:&f];
return f;
}
@end
:
CGFloat property = [self.object performFloatSelector:@selector(floatProperty)];
performSelector
方法只支持不返回任何值或返回一个对象的方法。这在它的文档中有说明:
对于返回非对象的方法,使用
NSInvocation
。
你的"hack"只是调用一个返回非对象的选择器的正确方法。
似乎适用于整型和布尔型方法的原因与方法返回值的方式有关。performSelector
的返回类型为id
,是对象指针类型。Integer-like值;其中包括NSInteger
、BOOL
和对象指针;通常在通用寄存器中返回,而浮点值通常在浮点寄存器中返回。编译器将始终从用于返回id
值的寄存器中加载结果,然后执行强制转换所需的任何操作。对于整型和布尔值,加载的值可能是正确的,在这些情况下强制转换是无操作的,所以一切(看起来)都可以工作。对于浮点值,加载的值是不正确的-它不是被调用方法返回的值,因为它在不同的寄存器中。
注意,使用performSelector
调用非对象、非void选择器会导致ARC下的内存问题——如果可能出现这种情况,编译器通常会发出警告。
NSInvocation
调用选择器的方法对非对象返回类型有效,其参数之一是方法签名本身。使用签名NSInvocation
可以确定返回值的类型及其返回方式,从而正确地将其返回给调用者。
HTH
如果您不想使用NSInvocation
,您可以直接使用objc_msgSend
函数(这是编译器将消息调用转换为的):
BOOL (*f)(id, SEL) = (BOOL (*)(id, SEL))objc_msgSend;
BOOL property = f(self.object, @selector(boolProperty));
NSInteger (*f)(id, SEL) = (NSInteger (*)(id, SEL))objc_msgSend;
NSInteger property = f(self.object, @selector(integerProperty));
CGFloat (*f)(id, SEL) = (CGFloat (*)(id, SEL))objc_msgSend;
CGFloat property = f(self.object, @selector(floatProperty));
注意,你必须将它强制转换为一个函数指针,该函数指针必须具有正确调用它的方法的确切签名;这里的方法没有参数,所以函数只有两个"隐藏"参数,self
和_cmd
。如果方法接受形参,则必须为函数指针类型添加更多形参。
另外,请注意,对于struct
返回,您需要使用objc_msgSend_stret
来代替objc_msgSend
,但其他所有内容与上面完全相同。