将完成 block 从一种方法传递到另一种执行长操作的方法是否正确:
- (void)searchWithCompletion:(void (^)(NSString* result))completion {
SearchManager *manager = [[SearchManager alloc] init]; // only one reference here
[manager search:^( NSString *name){ // it takes 3-5 seconds to execute
NSString *str = [NSString stringWithFormat@"Search result: %@", name];
completion (str);
}];
}
然后这样调用它
- (void)viewDidLoad
{
[super viewDidLoad];
[[MyManager sharedInstance] searchWithCompletion:^( NSString * result){
NSLog(@"foo");
}];
}
根据我的理解,manager
可以被销毁,因为没有对它的引用,并且它的完成 block 永远不会被调用。但经过一番尝试后,我发现这段代码总是可以正常工作,即使 searchWithCompletion
执行约 10 秒,我也可以看到“foo”。也许还有其他什么东西仍然持有经理
?为什么这样可以正常工作?
提前致谢。
最佳答案
如果没有看到搜索函数中的代码,就很难判断。然而,在调用完成 block 之前未释放管理器的最可能原因是词法作用域。这与在 searchWithCompletion 的第一行中创建 manager 后没有立即释放的原因相同。在您的情况下,如果管理器被释放,它将在您的 block 执行后,在 searchWithCompletion 方法调用的最后完成。您的搜索方法很可能类似于
- (void)search:(void (^)(NSString *name))completionBlock;
{
//do some searching
if (completionBlock)
{
completionBlock(@"joe");
}
}
这是假设您没有执行任何操作,例如将 block 存储为属性或使用某种其他类型的异步机制。但是,如果您记住, block 可以像任何其他变量一样被复制和传递。这意味着如果搜索将其传递给某个异步函数,那么该函数可以接管该 block 的所有权并允许管理器对象超出范围。
关于ios - 将 block 传递给方法,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24739312/