在Objective C中传递和调用动态块

mca*_*dre 5 objective-c objective-c-blocks

作为单元测试框架的一部分,我正在编写一个函数genArray,它将生成由传入的生成器块填充的NSArrays.因此[ObjCheck genArray: genInt]会产生一个随机整数的NSArray,[ObjCheck genArray: genChar]会产生一个随机字符的NSArray等.特别是,我在我的实现中遇到编译器错误,genArray并且genString是一个包装器[ObjCheck genArray: genChar].

我相信Objective C可以动态地操作块,但我没有正确的语法.

ObjCheck.m

+ (id) genArray: (id) gen {
    NSArray* arr = [NSMutableArray array];

    int len = [self genInt] % 100;

    int i;
    for (i = 0; i < len; i++) {
        id value = gen();

        arr = [arr arrayByAddingObject: value];
    }

    return arr;
}

+ (id) genString {
    NSString* s = @"";

    char (^g)() = ^() {
        return [ObjCheck genChar];
    };

    NSArray* arr = [self genArray: g];
    s = [arr componentsJoinedByString: @""];

    return s;
}
Run Code Online (Sandbox Code Playgroud)

当我尝试编译时,gcc抱怨它无法做到gen(),因为gen它不是一个函数.这是有道理的,因为gen它确实不是一个函数,而是id必须强制转换为函数.

但是当我重写使用的签名id^()代替时id,我也会遇到编译器错误.Objective C可以处理任意类型的块(genArray需要这个),还是太动态了?

Lil*_*ard 7

鉴于块是对象,您可以在块类型之间随时进行转换id,但是如果将块转换为错误的块类型并调用它,则会得到意外的结果(因为在运行时无法动态检查)块的"真实"类型是什么*).

BTW,id^()不是一种类型.你在考虑id(^)().这可能是您编译器错误的来源.您应该能够更新+genArray:以使用

id value = ((id(^)())(gen))();
Run Code Online (Sandbox Code Playgroud)

当然,这非常难看.

*实际上有一种方法,llvm将一个表示块类型的obj-c类型编码的字符串插入到块的内部结构中,但这是一个实现细节,依赖于您按顺序将块转换为其内部实现结构提取.