CoreFoundation:在后台线程中接收/处理通知

时间:2011-07-05 21:56:59

标签: multithreading macos background notifications runloop

我正在编写一个简单的应用程序,它应该能够使用Apple的CoreFoundation框架在后台线程中接收和处理通知。这是我正在努力实现的目标:

static void DummyCallback(CFNotificationCenterRef center,
              void *observer,
              CFStringRef name,
              const void *object,
              CFDictionaryRef userInfo) {
    printf("RECEIVED NOTIFICATION\n");
}

void *ThreadStart(void *arg) {
    CFNotificationCenterAddObserver(CFNotificationCenterGetDistributedCenter(),
                    NULL, 
                    &DummyCallback, 
                    NULL,
                    CFSTR("TEST_OBJECT"),
                    CFNotificationSuspensionBehaviorDeliverImmediately);

    printf("background thread: run run loop (should take 5 sec to exit)\n");
    int retval = CFRunLoopRunInMode(kCFRunLoopDefaultMode, 5, true);
    printf("background thread: exited from run loop (retval: %d)\n", retval);
    return NULL;
}

int main(int argc, char** argv) {
    pthread_t thread;
    int rc = pthread_create(&thread, NULL, &ThreadStart, NULL);
    assert(rc == 0);

    printf("main: sleep\n");
    sleep(10);
    printf("main: done sleeping\n");

    return 0;
}

如果我运行程序,我就得到

main: sleep
background thread: run run loop (should take 5 sec to exit)
background thread: exited from run loop (retval: 1)
main: done sleeping

问题是后台线程的运行循环立即退出(返回代码kCFRunLoopRunFinished而不是kCFRunLoopRunTimedOut),因为没有源/观察者/计时器。 CFNotificationCenterAddObserver仅在主线程的运行循环中注册自己,但不在我的后台线程中注册。

我需要其他东西的主线程,不能用它来运行它的运行循环。有没有办法让这个工作?也许通过使用后台线程的运行循环注册CFNotificationCenter?

提前致谢!

1 个答案:

答案 0 :(得分:1)

http://developer.apple.com/library/mac/#documentation/CoreFoundation/Reference/CFNotificationCenterRef/Reference/reference.html

中所述

第一次向分布式通知中心注册观察者时,通知中心会与系统范围的通知服务器建立连接,并将侦听端口置于当前线程运行循环的常用模式中。当传递通知时,它将在此初始线程上处理,即使接收通知的观察者在另一个线程上注册了通知。

因为加载的框架可能会在代码执行之前生成线程并添加自己的观察者,所以您无法确定哪个线程将接收分布式通知。如果需要控制哪个线程处理通知,则回调函数必须能够将通知转发到正确的线程。您可以使用CFMessagePort对象或自定义CFRunLoopSource对象将通知发送到正确的线程的运行循环。