将完成块从一个方法传递到执行长操作的另一个方法是否正确:
- (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
可以销毁,因为没有对它的引用,它的完成块永远不会被调用。但经过一些尝试,我发现这段代码总能正常工作,我可以看到'foo',即使searchWithCompletion
执行了~10秒。也许其他东西仍然存在manager
?为什么这样可以正常工作?
提前致谢。
答案 0 :(得分:1)
如果没有查看搜索功能中的代码,很难说清楚。然而,调用完成块之前管理器未被释放的最可能原因是因为词法范围。这与在managerWithCompletion的第一行中创建管理器后不立即被释放的原因相同。在你的情况下,如果正在释放manger,它将在你执行阻塞之后在searchWithCompletion方法调用的最后完成。您的搜索方法很可能类似于
- (void)search:(void (^)(NSString *name))completionBlock;
{
//do some searching
if (completionBlock)
{
completionBlock(@"joe");
}
}
这假设您没有做任何事情,比如将块存储为属性或使用其他类型的异步机制。但是,如果您记得可以像任何其他变量一样复制和传递块。这意味着如果搜索将其传递给某个异步函数,那么该函数可以接管块的所有权并允许管理器对象超出范围。
答案 1 :(得分:1)
从内存管理的角度来看,你所做的是正确的。在ARC和MRC中,manager
将在方法结束时释放,因为您不再使用它。
关于异步操作会发生什么,这完全取决于SearchManager
的设计方式,这是你无法从外部分辨出来的。 “搜索管理器”可能在内部执行某种异步操作,在完成后以某种方式回调给管理员(后者又调用完成处理程序),这意味着异步搜索操作必须对搜索管理器有某种引用(可能通过完成处理程序)。有两种可能性:
在你的情况下,第一个是真的。
答案 2 :(得分:0)
要告诉您更多信息,您应该发布SearchManager
的所有初始化相关代码。然而,
如果SearchManager
是单身,则在应用终止之前不会发布。
在软件工程中, singleton 模式是一种设计模式 这会将类的实例化限制为一个对象。
这可以解释为什么manager
对象没有被释放。
答案 3 :(得分:0)
根据实施情况,有两种可能的情况:
即使对象 manager 已被销毁,也可能正确调用完成处理程序。
对象 manager 可能会保留在方法search
中,并在调用完成处理程序之前(或之后)立即释放。
例如,NSOperation
的行为类似于#2。
对于案例#1可以想象,SearchManager
search
设置了一个带有完成处理程序的NSOperation
。一旦操作开始,就不再需要了它,并且可以被销毁。