iOS 64位模拟器上的va_list类型

时间:2015-03-17 09:26:52

标签: ios iphone 32bit-64bit variadic-functions

我有以下代码,它不适用于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;
}

1 个答案:

答案 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的类型,但我没有找到实现这一目标的解决方案。