__block
是一个存储限定符。它指定变量应该直接被块捕获,而不是复制它。如果您需要修改原始变量,这很有用,如下例所示
__block NSString *aString = @"Hey!";
void(^aBlock)() = ^{ aString = @"Hello!" }; // without __block you couldn't modify aString
NSLog(@"%@", aString); // Hey!
aBlock();
NSLog(@"%@", aString); // Hello!
在 ARC 中,这会导致变量自动保留,以便可以在块实现中安全地引用它。那么,在前面的示例中,当在块上下文中捕获时aString
会发送一条消息。retain
请注意,在 MRC(手动引用计数)中并非如此,其中变量被引用而不被保留。
将其标记为__weak
导致变量不被保留,因此块直接引用它但不保留它。这是潜在的危险,因为如果块的寿命比变量长,因为它将引用垃圾内存(并且可能会崩溃)。
这是clang doc中的相关段落:
在 Objective-C 和 Objective-C++ 语言中,我们允许对象类型变量的__weak
说明符。__block
[...]此限定符导致保留这些变量而不发送保留消息。如果块(或副本)超过此对象的生命周期,这会故意导致悬空指针。
最后,__block
可以用来避免强引用循环(也称为保留循环)的声明在 ARC 上下文中是完全错误的。由于在 ARC 中__block
导致变量被强引用,它实际上更有可能导致它们。
例如,在 MRC 中,此代码打破了保留周期
__block typeof(self) blockSelf = self; //this would retain self in ARC!
[self methodThatTakesABlock:^ {
[blockSelf doSomething];
}];
而要在 ARC 中获得相同的结果,您通常会这样做
__weak typeof(self) weakSelf = self;
[self methodThatTakesABlock:^ {
[weakSelf doSomething];
}];