Objective-C块,递归失败

苏家伙,

我正在尝试做一个自我调用函数,但是把所有东西放在一个块上,

正如您所看到的,以下函数旨在被无限次调用(直到arcrandom返回低于50的数字),并且您应该期望作为输出可变数量的“RUNNING”消息,具体取决于偶然性.

void (^_test_closure)(void) = ^ {
    NSLog(@"RUNNING");
    if(arc4random() % 100 > 50) {
        _test_closure();
    }
};

_test_closure();

但是,在运行它时,我得到一个EXC_BAD_ACCESS错误,我发现的原因是,当代码试图在闭包内调用_test_closure时,它基本上指向无处.

有谁知道如何使上述代码工作?

解决方法

递归和块是棘手的.因为一个块捕获传入的所有变量,所以变量_test_closure尚未初始化(并且clang应该给你一个警告:

Block pointer variable ‘_test_closure’ is uninitialized when captured by block

).

有几种方法可以解决这个问题,但最明显的方法是最简单的是让块本身成为一个__block变量(@ H2CO3所说的).这允许块几乎是弱链接的,因此当您再次调用它时,它会被正确初始化.

你有另一个选择是使块成为全局或静态,如下所示:

// outside of 'main',thus being a global variable
void (^blockRecurse)(int) = ^(int level) {
    if (level < 0)
        return;
    NSLog(@"Level: %i",level);
    blockRecurse(--level);
};

int main()
{
    @autoreleasepool {
        blockRecurse(10);
    }
}

这意味着它不会被块捕获,而是引用全局/静态变量,可以由所有代码同等地更改.

相关文章

本程序的编译和运行环境如下(如果有运行方面的问题欢迎在评...
水了一学期的院选修,万万没想到期末考试还有比较硬核的编程...
补充一下,先前文章末尾给出的下载链接的完整代码含有部分C&...
思路如标题所说采用模N取余法,难点是这个除法过程如何实现。...
本篇博客有更新!!!更新后效果图如下: 文章末尾的完整代码...
刚开始学习模块化程序设计时,估计大家都被形参和实参搞迷糊...