将块传递给方法

时间:2014-07-14 14:48:41

标签: ios objective-c objective-c-blocks

将完成块从一个方法传递到执行长操作的另一个方法是否正确:

- (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?为什么这样可以正常工作?

提前致谢。

4 个答案:

答案 0 :(得分:1)

如果没有查看搜索功能中的代码,很难说清楚。然而,调用完成块之前管理器未被释放的最可能原因是因为词法范围。这与在managerWithCompletion的第一行中创建管理器后不立即被释放的原因相同。在你的情况下,如果正在释放manger,它将在你执行阻塞之后在searchWithCompletion方法调用的最后完成。您的搜索方法很可能类似于

- (void)search:(void (^)(NSString *name))completionBlock;
{
    //do some searching

    if (completionBlock)
    {
        completionBlock(@"joe");
    }
}

这假设您没有做任何事情,比如将块存储为属性或使用其他类型的异步机制。但是,如果您记得可以像任何其他变量一样复制和传递块。这意味着如果搜索将其传递给某个异步函数,那么该函数可以接管块的所有权并允许管理器对象超出范围。

答案 1 :(得分:1)

从内存管理的角度来看,你所做的是正确的。在ARC和MRC中,manager将在方法结束时释放,因为您不再使用它。

关于异步操作会发生什么,这完全取决于SearchManager的设计方式,这是你无法从外部分辨出来的。 “搜索管理器”可能在内部执行某种异步操作,在完成后以某种方式回调给管理员(后者又调用完成处理程序),这意味着异步搜索操作必须对搜索管理器有某种引用(可能通过完成处理程序)。有两种可能性:

  • 搜索操作(或管理员传递给它的完成块)具有对搜索管理器的强引用。在这种情况下,搜索管理器将自动保持活动状态,直到操作完成。
  • 搜索操作(或管理员传递给它的完成块)对搜索管理器的引用很弱。在这种情况下,如果你没有保留对搜索管理器的引用,它将被解除分配,并且搜索操作将被有效地“取消”(好吧,底层操作可能仍然完成,但它将回调为零) 'ed reference,所以它可能不会做任何事情)。如果是这种情况,则必须保留对它的引用。

在你的情况下,第一个是真的。

答案 2 :(得分:0)

要告诉您更多信息,您应该发布SearchManager的所有初始化相关代码。然而, 如果SearchManager是单身,则在应用终止之前不会发布。

  

在软件工程中, singleton 模式是一种设计模式   这会将类的实例化限制为一个对象

这可以解释为什么manager对象没有被释放。

答案 3 :(得分:0)

根据实施情况,有两种可能的情况:

  1. 即使对象 manager 已被销毁,也可能正确调用完成处理程序。

  2. 对象 manager 可能会保留在方法search中,并在调用完成处理程序之前(或之后)立即释放。

  3. 例如,NSOperation的行为类似于#2。

    对于案例#1可以想象,SearchManager search设置了一个带有完成处理程序的NSOperation。一旦操作开始,就不再需要了它,并且可以被销毁。