在主线程上运行NSURLSession完成处理程序

时间:2013-11-18 16:24:30

标签: ios multithreading cocoa ios7

我使用NSURLSession来获取填充TableView的值。我正在完成处理程序中更新TableView,但使用[[NSThread currentThread] isMainThread]向我显示完成处理程序未在主线程中运行。由于我只应该从主线程更新UI,我知道这是不正确的。有没有办法从完成处理程序触发主线程上的操作?或者使用NSURLSession是错误的方法来解决这个问题?

NSURLSession *session = [NSURLSession sharedSession];
[[session dataTaskWithURL:[NSURL URLWithString:@"http://myurl"]
        completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
            NSError *jsonError = nil;
            NSArray* jsonUsers = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError];
            if (jsonError) {
                NSLog(@"error is %@", [jsonError localizedDescription]);
                // Handle Error and return
                return;
            }
            self.userArray = jsonUsers;
            [self.userTableView reloadData];
            if ([[NSThread currentThread] isMainThread]){
                NSLog(@"In main thread--completion handler");
            }
            else{
                NSLog(@"Not in main thread--completion handler");
            }
}] resume];

6 个答案:

答案 0 :(得分:41)

是的,只需使用GCD

调度您的主要内容
 NSURLSession *session = [NSURLSession sharedSession];
    [[session dataTaskWithURL:[NSURL URLWithString:@"http://myurl"]
            completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
                NSError *jsonError = nil;
                NSArray* jsonUsers = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError];
                if (jsonError) {
                    NSLog(@"error is %@", [jsonError localizedDescription]);
                    // Handle Error and return
                    return;
                }

                dispatch_async(dispatch_get_main_queue(), ^{
                    self.userArray = jsonUsers;
                    [self.userTableView reloadData];
                    if ([[NSThread currentThread] isMainThread]){
                        NSLog(@"In main thread--completion handler");
                    }
                    else{
                        NSLog(@"Not in main thread--completion handler");
                    }
                });

            }] resume];

答案 1 :(得分:18)

@ graver的答案很好。这是另一种方法:

NSURLSession* session = [NSURLSession sessionWithConfiguration:[NSURLSessionConfiguration defaultSessionConfiguration]
                                                          delegate:nil
                                                     delegateQueue:[NSOperationQueue mainQueue]];
[[session dataTaskWithURL:[NSURL URLWithString:@"http://myurl"]
        completionHandler:^(NSData *data, NSURLResponse *response, NSError *error) {
            NSError *jsonError = nil;
            NSArray* jsonUsers = [NSJSONSerialization JSONObjectWithData:data options:0 error:&jsonError];
            if (jsonError) {
                NSLog(@"error is %@", [jsonError localizedDescription]);
                // Handle Error and return
                return;
            }

            self.userArray = jsonUsers;
            [self.userTableView reloadData];
            if ([[NSThread currentThread] isMainThread]){
                NSLog(@"In main thread--completion handler");
            }
            else{
                NSLog(@"Not in main thread--completion handler");
            }
        }] resume];

这样您就可以创建一个会话来调用主线程上的完成块和任何委托方法。您可能会发现这更美观,但您确实失去了在后台运行“艰苦工作”的优势。

答案 2 :(得分:3)

这是从块和完成处理程序更新UI的最佳方法,也是在您没有确定运行代码的哪个线程时。

static void runOnMainThread(void (^block)(void))
{
    if (!block) return;

    if ( [[NSThread currentThread] isMainThread] ) {
        block();
    } else {
        dispatch_async(dispatch_get_main_queue(), block);
    }
}

这是一个静态方法,它将有一个块,并将在主线程上运行,它将像

一样
runOnMainThread(^{
           // do things here, it will run on main thread, like updating UI

        });

答案 3 :(得分:3)

你可以试试这个:

[self.userTableView performSelectorOnMainThread:@selector(reloadData) withObject:nil waitUntilDone:NO];

答案 4 :(得分:0)

在完成后发送一个通知,表视图控制器将观察该通知,然后执行reloadData;

这样做的好处是,如果您稍后将此下载代码移至单独的类,例如一个模型对象,然后不需要进行任何更改,并且代码可以在其他项目中重用而无需进行更改

答案 5 :(得分:0)

Swift 3.1

DispatchQueue.main.async {
    tableView.reloadData()
}