0

我试图用来CFRunLoopRunInMode()避免在[AFHTTPClient getPath:...]完成块中返回。

我的代码如下所示:

NSLog(@"start");
__block BOOL someCondition = NO;
AFHTTPClient *client = [[AFHTTPClient alloc] initWithBaseURL:[NSURL URLWithString:@"http://domain.com"]];
[client getPath:@"my/path" parameters:nil success:^(AFHTTPRequestOperation *operation, id responseObject) {
    NSLog(@"success");
} failure:^(AFHTTPRequestOperation *operation, NSError *error) {
    dispatch_async(dispatch_get_main_queue(), ^{
        NSLog(@"async log");
        someCondition = YES;
    });
    while (!someCondition) {
        CFRunLoopRunInMode(kCFRunLoopDefaultMode, 0.5, YES);
    }
    NSLog(@"failure");
}];

我预计输出是:

start
async log
failure

但相反,我只得到:

start

CFRunLoopRunInMode()返回kCFRunLoopRunHandledSource,但调度队列从不执行提交的块。如果我在完成块之外运行相同的代码,则输出符合预期。

我无法弄清楚为什么从完成块运行时调度队列没有被处理。

有人可以解释一下为什么会这样吗?

4

2 回答 2

1

我无法弄清楚为什么从完成块运行时调度队列没有被处理。

因为您没有“运行”调度队列(没有“运行”调度队列之类的东西)。你运行了运行循环。调度队列是相关的,但是是另一回事。在串行队列(如 main)上一次只能运行一个块。在您的块完成之前,不能安排其他块。GCD 中没有 API 可以规避这一点。这通常是一件非常好的事情,因为它为运行循环操作中并不总是存在的队列操作提供了确定性。

于 2013-04-15T21:35:23.463 回答
1

AFNetwork 的成功和失败块都调度在主队列中,你在失败块中写入的异步块也调度在主队列中。

在故障块完成之前,您的 GCD 同步块将运行。因为'while'条件永远不会为假,所以失败块永远不会完成,也永远不会有机会被执行。

于 2014-09-10T09:35:15.420 回答