我有一个NSObject类别来在一段时间后执行块。在这种情况下,我是否需要弱自我?
__weak ViewController *weakSelf = self;
[self runBlockAfterDelay:0.6 block:^{
weakSelf.someview = ...
}];
// Category
- (void)runBlockAfterDelay:(NSTimeInterval)delay block:(void (^)(void))block {
dispatch_async(dispatch_get_main_queue(), ^{
[self performSelector:@selector(executeBlockAfterDelay:) withObject:[block copy] afterDelay:delay];
});
}
- (void)executeBlockAfterDelay:(void(^)(void))block
{
if (block)
block();
}
答案 0 :(得分:6)
你不仅不需要它(自己没有对块的强引用,所以没有循环),如果你包含它,它可能会给你的程序带来严重的错误。具体来说,在块运行之前没有任何东西可以防止“self”被释放(因为弱引用的全部意义在于它不会阻止事件被释放)。
答案 1 :(得分:4)
只是为@ Catfish_Man的精彩描述添加更多细节:
您可以在阻止之前弱化自我,然后在阻止内部再次强化它以确保它不会在意外时间释放。在这种情况下,应该确保在释放self之后不执行该块。这很难,所以只要没有创建周期,保持强大是一个更安全的选择。 (如果自己持有对块的引用,则会创建循环。)
仅供参考:如果您开始考虑信号而不是程序排序,Reactive Cocoa的最终结果可能会更令人满意:
NSTimeInterval delay = 0.3;
// The activator signal that fires after the delay and completes after it.
RACSignal *delayedActivator = [[RACSignal interval:delay] take:1];
@weakify(self)
// The command that we want to execute after the delay.
RACCommand *blockToExecute = [RACCommand command];
[blockToExecute addSignalBlock:^RACSignal *(id value) {
@strongify(self)
self.whatever
}];
// Wire up the command on the signal.
[delayedActivator executeCommand:blockToExecute];
乍一看可能看起来有点神秘,但信号可以让你的生活变得更轻松。特别是因为如果信号与自我的生命周期有关,例如它是一个属性,它将在self释放时释放,这样可以确保在不需要时不执行块。在这种情况下,需要弱化和强化。