我正在尝试在 iOS5 中同步以下代码:
- 一个对象有一个方法,它发出一个 HTTP 请求,从中获取一些数据,包括一个图像的 URL
- 数据到达后,文本数据用于填充 CoreData 模型
- 同时,异步调度第二个线程来下载图像;当图像已经缓存并在 CoreData 模型中可用时,该线程将通过 KVO 向 viewController 发出信号。
- 由于图像下载需要一段时间,我们立即将具有除图像之外的所有属性的 CoreData 对象返回给调用者。
- 此外,当第二个线程完成下载时,可以保存 CoreData 模型。
这是(简化的)代码:
- (void)insideSomeMethod
{
[SomeHTTPRequest withCompletionHandler:
^(id retrievedData)
{
if(!retrievedData)
{
handler(nil);
}
// Populate CoreData model with retrieved Data...
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
NSURL* userImageURL = [NSURL URLWithString:[retrievedData valueForKey:@"imageURL"]];
aCoreDataNSManagedObject.profileImage = [NSData dataWithContentsOfURL:userImageURL];
});
handler(aCoreDataNSManagedObject);
[self shouldCommitChangesToModel];
}];
}
- (void)shouldCommitChangesToModel
{
dispatch_barrier_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
NSError *error = nil;
if(![managedObjectContext save:&error])
{
// Handle error
}
});
}
但实际情况是基于屏障的保存块总是在图像加载块之前执行。那是,
dispatch_barrier_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
NSError *error = nil;
if(![managedObjectContext save:&error])
{
// Handle error
}
});
之前执行:
dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_HIGH, 0), ^{
NSURL* userImageURL = [NSURL URLWithString:[retrievedData valueForKey:@"imageURL"]];
aCoreDataNSManagedObject.profileImage = [NSData dataWithContentsOfURL:userImageURL];
});
所以很明显我并没有真正在屏障之前调度图像加载块,或者屏障会等到图像加载块完成后再执行(这是我的意图)。
我究竟做错了什么?如何确保图像加载块在屏障块之前排队?