苏家伙,
我正在尝试做一个自我调用的函数,但是把所有东西放在一个块上,
正如您所看到的,以下函数旨在被无限次调用(直到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);
- }
- }
这意味着它不会被块捕获,而是引用全局/静态变量,可以由所有代码同等地更改.