我有以下代码,它不适用于64位iOS模拟器。它在以下行EXC_BAD_ACCESS
[invocation setArgument:args atIndex:index];
崩溃
但是这段代码适用于所有32位和64位iPhone和iPad设备,甚至适用于32位模拟器。
到目前为止我发现,char*
的演员可能是错误的来源。除了64位iOS模拟器之外,va_list
似乎是char*
类型。
我如何避免此错误?这会导致设备出现问题吗?
+(NSInvocation*)invocationWithTarget:(id)target selector:(SEL)aSelector
retainArguments:(BOOL)retainArguments, ...;
{
va_list ap;
va_start(ap, retainArguments);
char* args = (char*)ap;
NSMethodSignature* signature = [target methodSignatureForSelector:aSelector];
NSInvocation* invocation = [NSInvocation invocationWithMethodSignature:signature];
if (retainArguments) {
[invocation retainArguments];
}
[invocation setTarget:target];
[invocation setSelector:aSelector];
for (int index = 2; index < [signature numberOfArguments]; index++) {
const char *type = [signature getArgumentTypeAtIndex:index];
NSUInteger size, align;
NSGetSizeAndAlignment(type, &size, &align);
NSUInteger mod = (NSUInteger)args % align;
if (mod != 0) {
args += (align - mod);
}
[invocation setArgument:args atIndex:index];
args += size;
}
va_end(ap);
return invocation;
}
答案 0 :(得分:2)
我认为问题出现在args += (align - mod);
,va_arg
如何工作尚不清楚,两个版本的编译器之间可能有所不同。我建议使用va_arg
而不是自己移动args
的指针。
这样你必须使用Objective-C对象作为参数,因为我们假设它们的类型是id
。
+ (NSInvocation*)invocationWithTarget:(id)target selector:(SEL)aSelector
retainArguments:(BOOL)retainArguments, ...
{
va_list ap;
va_start(ap, retainArguments);
NSMethodSignature* signature = [target methodSignatureForSelector:aSelector];
NSInvocation* invocation = [NSInvocation invocationWithMethodSignature:signature];
if (retainArguments) {
[invocation retainArguments];
}
[invocation setTarget:target];
[invocation setSelector:aSelector];
for (int index = 2; index < [signature numberOfArguments]; index++) {
id argument = va_arg(ap, id) ;
[invocation setArgument:&argument atIndex:index];
}
va_end(ap);
return invocation;
}
我相信我们可以更好地使用const char *type = [signature getArgumentTypeAtIndex:index];
并使用某种方法来解码可以传递给va_arg
的类型,但我没有找到实现这一目标的解决方案。