3

効果がないように見えるケースがありCFRunLoopWakeUpます。セットアップは次のとおりです。

メインスレッドにない「典型的な」whileループがあり、いくつかの作業が完了するのを待ちます。

- (void)someFunc
{
    self.runLoop = CFRunLoopGetCurrent();
    NSLog(@"Pre loop.");
    while (!self.completed)
    {
        NSLog(@"In loop.");
        NSAutoreleasePool * pool = [[NSAutoreleasePool alloc] init];
        [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        [pool release];
    }
    NSLog(@"Post loop.");
}

何らかの作業が完了するまで待機するコールバック関数があります。これもメインスレッドから呼び出されません。

- (void)callback
{
    NSLog(@"Work completed.");
    self.completed = YES;

    // I've checked that CFRunLoopIsWaiting(self.runLoop) here returns true
    CFRunLoopWakeUp(self.runLoop); // Should wake up the waiting run loop, but doesn't!
}

コールバックが呼び出されますが、何らかの理由でCFRunLoopWakeUp何もしていないようです。明らかな何かが欠けていますか?ここでいくつかの深いスレッド化の問題が発生していますか? ありがとう!

4

2 に答える 2

0

まず、私はあなたの問題を再現することはできません。私は次のようにGCDで構築しています:

int main (int argc, const char *argv[])
{
  @autoreleasepool {
    __block BOOL completed = NO;
    __block CFRunLoopRef runLoop;

    dispatch_queue_t queue1 = dispatch_queue_create("first", 0);
    dispatch_queue_t queue2 = dispatch_queue_create("second", 0);

    dispatch_async(queue1, ^{
      runLoop = CFRunLoopGetCurrent();
      NSLog(@"Pre loop.");
      while (!completed)
      {
        NSLog(@"In loop.");
        @autoreleasepool {
          [[NSRunLoop currentRunLoop] runMode:NSDefaultRunLoopMode beforeDate:[NSDate distantFuture]];
        }
      }
      NSLog(@"Post loop.");
    });

    double delayInSeconds = 2.0;
    dispatch_time_t popTime = dispatch_time(DISPATCH_TIME_NOW, (unsigned)(delayInSeconds * NSEC_PER_SEC));
    dispatch_after(popTime, queue2, ^(void) {
      NSLog(@"Work completed.");
      completed = YES;

      // I've checked that CFRunLoopIsWaiting(self.runLoop) here returns true
      CFRunLoopWakeUp(runLoop); // Should wake up the waiting run loop, but doesn't!

    });

    dispatch_sync(queue1, ^{});
    dispatch_sync(queue2, ^{});
    dispatch_release(queue1);
    dispatch_release(queue2);
  }
  return 0;
}

問題を実証するより簡単なプログラムを作成できますか?

私が試みる他のことは、主に問題を絞り込むためのデバッグ目的です。

  • CFRunLoopRunInMode()ではなくに切り替えますruninMode:beforeDate:。それらはわずかに異なります。
  • CFRunLoopStop()ではなくに切り替えますCFRunLoopWakeUp()

そしてもちろん、それがself.runLoop実際にあなたがそう思うランループを指していることを確認してください!

于 2012-06-28T05:06:58.510 に答える
-1

この男が説明しているように、ソースを追加することで CFRunLoopWakeUp を機能させることができました

于 2012-06-26T18:24:18.067 に答える