我必须从我的RestAPI同步一堆信息。我必须完成6个RestAPI调用才能完成工作。我设计了带块的API调用,并返回NSError (如果有的话)。其中3次调用应该执行嵌套调用,因为第一次调用给其他调用并允许执行,而其他3次调用可以独立运行。由于提高了网络性能,我将同步调用设计为:
最后一个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];
}
}];
}怎么了?
发布于 2014-08-29 22:18:15
问题是NSBlockOperation是用于同步块的。一旦它的块完成执行,它将是finished。如果其块触发异步方法,这些方法将独立运行.
例如,当您的syncUserInfoOperation块被执行时,它会触发[dataSync syncUserInfo:...],然后认为自己已经完成;它不会等待任何完成处理程序触发,或者诸如此类的事情。
一个很好的解决方案是创建您自己的NSOperation子类。您可能希望为每个数据同步类型创建一个,以便更容易地设置依赖关系等,但这取决于您。您可以阅读有关如何执行这里的所有内容(请务必阅读关于“为并发执行配置操作”的部分)。
您还可以创建一个通用的NSOperation子类,该子类接受一个可以异步运行的块。这方面的主要问题是,它使得处理诸如取消操作这样的事情变得更加困难,而您可能仍然想要这样的操作。
https://stackoverflow.com/questions/25573374
复制相似问题