3

我正在编写一个简单的应用程序,它应该能够使用 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 ?

提前致谢!

4

1 回答 1

1

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

观察者第一次注册到分布式通知中心时,通知中心会创建与系统范围的通知服务器的连接,并将监听端口放入当前线程运行循环的公共模式中。当通知被传递时,它会在这个初始线程上进行处理,即使接收通知的观察者在不同的线程上注册了通知。

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

于 2011-07-06T19:17:57.157 回答