6

我必须从我的 RestAPI 同步一堆信息。我必须进行 6 次 RestAPI 调用才能完成工作。我用 Blocks 设计了 ​​API 调用,如果有的话返回 NSError。其中 3 个调用应该嵌套执行,因为第一个调用向其他调用提供信息并允许执行,而其他 3 个调用可以独立运行。由于提高网络性能,我设计了我的同步调用如下:

  • 1 NSBlockOperation 包含前嵌套的 3 个块;
  • 1个包含其他三个块的NSBlockOperation;
  • 1 NSBlockOperation 我用作“信号”并告诉我所有工作何时完成。

最后一个 NSBlockOperation 依赖于前两个 NSBlockOperation。

我还有一个 NSOperationQueue 包含所有三个 NSBlockOperation ,其中信号量 NSBlockOperation 作为队列中的最后一个添加。我要实现的结果是:前两个块同时调用,当它们的工作完成时,调用信号量 NSBlockOperation 并将控件返回给提供 UIAlertMessage 的用户。

结果与前面解释的不一样:控件在没有等待syncAllBlocksInformation块结束的情况下返回。

在包含 NSBlockOperation 的代码下方:

-(void)syncAllBlocksInformation:(void(^)(NSError *error))completion{

__block NSError *blockError = nil;

NSOperation *syncUserInfoOperation = [NSBlockOperation blockOperationWithBlock:^{
    [dataSync syncUserInfo:tfMail.text password:tfPassword.text completion:^(NSError *error, NSNumber *idUser) {
        if(!error){
            [dataSync syncUserfilesInfo:idUser completion:^(NSError *error) {
                if(!error){
                    [dataSync syncUserBookings:^(NSError *error) {
                        if(error){
                            blockError = error;
                        }
                    }];
                }
                else{
                    blockError = error;
                }
            }];

        }
        else{
            blockError = error;
        }
    }];
}];



NSBlockOperation *otherSyncOperations = [NSBlockOperation blockOperationWithBlock:^{
    [dataSync syncNewsInfo:^(NSError *error) {
        if(error){
            blockError = error;
            NSLog(@"error %@",error);
        }
    }];

}];

[otherSyncOperations addExecutionBlock:^{
    [dataSync syncLocationsInfo:^(NSError *error) {
        if(error){
            blockError = error;
            NSLog(@"error %@",error);
        }
    }];

}];

[otherSyncOperations addExecutionBlock:^{
    [dataSync syncExoticAnimalTypesAndAnimals:^(NSError *error) {
        if(error){
            blockError = error;
            NSLog(@"error %@",error);
        }
    }];
}];


NSOperation *completionOperation = [NSBlockOperation blockOperationWithBlock:^{
    NSLog(@"END");
}];

[completionOperation setCompletionBlock:^{
    NSLog(@"Syc isEx %i",syncUserInfoOperation.isExecuting);
    NSLog(@"other isEx %i",otherSyncOperations.isExecuting);
    completion(blockError);
}];

NSOperationQueue *opQueue = [NSOperationQueue new];

[completionOperation addDependency:syncUserInfoOperation];
[completionOperation addDependency:otherSyncOperations];

[opQueue addOperation:syncUserInfoOperation];
[opQueue addOperation:otherSyncOperations];
[opQueue addOperation:completionOperation];

}

在这里,调用上述块的代码:

-(IBAction)login:(id)sender{

[self dismissKeyboardOpened:nil];

hud=[MBProgressHUD showHUDAddedTo:self.view animated:YES];
[hud setLabelText:NSLocalizedString(@"login_hud_message", login_hud_message )];
[hud setMode:MBProgressHUDModeIndeterminate];

[self showHudAndNetworkActivity:YES];

[self syncAllBlocksInformation:^(NSError *error) {

    [self showHudAndNetworkActivity:NO];

    if(!error){
        NSLog(@"End LOGIN");
        [self showAlert:@"Login" message:@"Login OK" dismiss:YES];
    }
    else{
        [self showAlert:@"Error" message:@"Login NO" dismiss:NO];
    }

}];
}

怎么了 ?

4

1 回答 1

5

问题在于同步NSBlockOperation。它将在其块完成执行后立即执行。如果它的块触发异步方法,这些将独立运行。finished

例如,当你syncUserInfoOperation的 ' 块被执行时,它会触发[dataSync syncUserInfo:...],然后认为自己完成了;它不会等待任何完成处理程序触发或类似的事情。

一个很好的解决方案是创建自己的NSOperation子类。您可能希望为每种数据同步类型创建一个,以便更轻松地设置依赖项等,但这取决于您。您可以在此处阅读有关如何执行此操作的所有信息(请务必阅读“为并发执行配置操作”部分)。

您还可以创建一个通用NSOperation子类,该子类采用可以异步运行的块。这样做的主要问题是它使处理诸如取消操作之类的事情变得更加困难,而您可能仍然想要这样做。

于 2014-08-29T22:18:15.127 回答