How to wait for a thread to finish in Objective-C

前端 未结 7 1946
长情又很酷
长情又很酷 2020-12-23 15:28

I\'m trying to use a method from a class I downloaded somewhere. The method executes in the background while program execution continues. I do not want to allow program exec

相关标签:
7条回答
  • 2020-12-23 15:57

    My technique is to check if the task can run (if the background thread is finished), and if it can't run then I use GCD to try again after a delay:

    - (void)doSomething
    {
      if (/* is the other thread done yet? */) {
        double delayInSeconds = 2.0;
        dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(delayInSeconds * NSEC_PER_SEC));
        dispatch_after(popTime, dispatch_get_main_queue(), ^(void){
          [self doSomething];
        });
        return;
      }
    
      // do something
    }
    
    0 讨论(0)
  • 2020-12-23 16:04

    Here's another way to do it using GCD:

    
    
    - (void)main
    {
        [self doStuffInOperations];
    }
    
    - (void)doStuffInGCD
    {
        dispatch_group_t d_group = dispatch_group_create();
        dispatch_queue_t bg_queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    
        dispatch_group_async(d_group, bg_queue, ^{
            [self doSomething:@"a"];
        });
    
        dispatch_group_async(d_group, bg_queue, ^{
            [self doSomething:@"b"];
        });
    
        dispatch_group_async(d_group, bg_queue, ^{
            [self doSomething:@"c"];
        });
    
    
        // you can do this to synchronously wait on the current thread:
        dispatch_group_wait(d_group, DISPATCH_TIME_FOREVER);
        dispatch_release(d_group);
        NSLog(@"All background tasks are done!!");
    
    
        // ****  OR  ****
    
        // this if you just want something to happen after those are all done:
        dispatch_group_notify(d_group, dispatch_get_main_queue(), ^{
            dispatch_release(d_group);
            NSLog(@"All background tasks are done!!");        
        });
    }
    
    - (void)doSomething:(id)arg
    {
        // do whatever you want with the arg here 
    }
    
    0 讨论(0)
  • 2020-12-23 16:08

    Several technical ways to synchronize multi-threads, such as NSConditionLock(mutex-lock), NSCondition(semaphore)。But they are common programming knowledge for other languages(java...) besides objective-c. I prefer to introduce run loop(special in Cocoa) to implement thread join:

    NSThread *A; //global
    A = [[NSThread alloc] initWithTarget:self selector:@selector(runA) object:nil]; //create thread A
    [A start];
    
    - (void)runA    
    {
      [NSThread detachNewThreadSelector:@selector(runB) toTarget:self withObject:nil]; //create thread B    
      while (1)    
      {    
        if ([[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]]) //join here, waiting for thread B    
        {    
          NSLog(@"thread B quit...");    
          break;    
        }    
      }    
    }
    
    - (void)runB    
    {    
      sleep(1);    
      [self performSelector:@selector(setData) onThread:A withObject:nil waitUntilDone:YES modes:@[NSDefaultRunLoopMode]];    
    }
    
    0 讨论(0)
  • 2020-12-23 16:09

    For such cases I usually using NSCondition class.

    //this method executes in main thread/queue
    - (void)waitForJob
    {
        id __weak selfWeak = self;
        NSCondition *waitHandle = [NSCondition new];
        [waitHandle lock];
        dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_BACKGROUND, 0), ^{
            [selfWeak doSomethingLongtime];
            [waitHandle signal];
        });
        //waiting for background thread finished
        [waitHandle waitUntilDate:[NSDate dateWithTimeIntervalSinceNow:60]];
    }
    
    0 讨论(0)
  • My first inclination is to not do what you are suggesting. The technique I've used before is to give the thread a selector to a method in the originating object (which is on the main thread). When the second thread is started, the main thread keeps executing, but puts up a busy indicator of some sort on the display. This allows user interaction to continue if required.

    When the second thread ends, just before it shuts down, it calls the selector on the main thread. The method that the selector references then removes the busy indicator from the display and tells the main thread to update, picking up whatever data the second thread has generated.

    I've used this successfully for an app which accesses a web service (on the second thread) and then updates the display once data is returned without locking it. This makes the user experience much nicer.

    0 讨论(0)
  • 2020-12-23 16:12

    Use an NSOperationQueue, like this
    (from memory so forgive any minor errors -- you'll get the basic idea):

    
    // ivars
    NSOperationQueue *opQueue = [[NSOperationQueue alloc] init];
    // count can be anything you like
    [opQueue setMaxConcurrentOperationCount:5];
    
    - (void)main
    {
        [self doStuffInOperations];
    }
    
    // method
    - (void)doStuffInOperations
    {
        // do parallel task A
        [opQueue addOperation:[[[NSInvocationOperation alloc] initWithTarget:self selector:@selector(doSomething:) object:@"a"] autorelease]];
    
        // do parallel task B
        [opQueue addOperation:[[[NSInvocationOperation alloc] initWithTarget:self selector:@selector(doSomething:) object:@"b"] autorelease]];
    
        // do parallel task C
        [opQueue addOperation:[[[NSInvocationOperation alloc] initWithTarget:self selector:@selector(doSomething:) object:@"c"] autorelease]];
    
    
        [opQueue waitUntilAllOperationsHaveFinished];
    
        // now, do stuff that requires A, B, and C to be finished, and they should be finished much faster because they are in parallel.
    }
    
    - (void)doSomething:(id)arg
    {
        // do whatever you want with the arg here 
        // (which is in the background, 
        // because all NSOperations added to NSOperationQueues are.)
    }
    
    
    
    0 讨论(0)
提交回复
热议问题