Charles 的回答是正确的,但他的警告(“确保只从处理程序中调用可重入函数”)是一个极端限制。可以使用kqueue 和CFFileDescriptor 将信号处理重定向到更安全的环境。
Technical Note TN2050: Observing Process Lifetimes Without Polling 是一个不同的主题,但说明了该技术。在那里,Apple 以这种方式描述了 Charles 的警告:
由于古怪的执行,监听信号可能会很棘手
与信号处理程序相关的环境。具体来说,如果你
安装一个信号处理程序(使用signal 或sigaction),你必须非常
小心你在那个处理程序中所做的事情。很少有功能是安全的
从信号处理程序调用。例如,分配是不安全的
内存使用malloc!
对信号处理程序安全的函数(异步信号
安全函数)列在sigaction man page。
在大多数情况下,您必须采取额外的步骤来重定向传入信号
到一个更明智的环境。
我从那里获取了代码插图并对其进行了修改以处理SIGINT。抱歉,这是 Objective-C。这是一次性设置代码:
// Ignore SIGINT so it doesn't terminate the process.
signal(SIGINT, SIG_IGN);
// Create the kqueue and set it up to watch for SIGINT. Use the
// EV_RECEIPT flag to ensure that we get what we expect.
int kq = kqueue();
struct kevent changes;
EV_SET(&changes, SIGINT, EVFILT_SIGNAL, EV_ADD | EV_RECEIPT, NOTE_EXIT, 0, NULL);
(void) kevent(kq, &changes, 1, &changes, 1, NULL);
// Wrap the kqueue in a CFFileDescriptor. Then create a run-loop source
// from the CFFileDescriptor and add that to the runloop.
CFFileDescriptorContext context = { 0, self, NULL, NULL, NULL };
CFFileDescriptorRef kqRef = CFFileDescriptorCreate(NULL, kq, true, sigint_handler, &context);
CFRunLoopSourceRef rls = CFFileDescriptorCreateRunLoopSource(NULL, kqRef, 0);
CFRunLoopAddSource(CFRunLoopGetCurrent(), rls, kCFRunLoopDefaultMode);
CFRelease(rls);
CFFileDescriptorEnableCallBacks(kqRef, kCFFileDescriptorReadCallBack);
CFRelease(kqRef);
以下是实现上述sigint_handler 回调的方法:
static void sigint_handler(CFFileDescriptorRef f, CFOptionFlags callBackTypes, void *info)
{
struct kevent event;
(void) kevent(CFFileDescriptorGetNativeDescriptor(f), NULL, 0, &event, 1, NULL);
CFFileDescriptorEnableCallBacks(f, kCFFileDescriptorReadCallBack);
// You've been notified!
}
请注意,此技术要求您在线程上运行设置代码,只要您有兴趣处理 SIGINT(可能是应用程序的生命周期)并服务/运行其运行循环,该线程就会一直存在。系统为其自身目的创建的线程(例如为 Grand Central Dispatch 队列提供服务的线程)不适合此目的。
应用程序的主线程将工作,您可以使用它。 但是,如果主线程被锁定或变得无响应,那么它就不会为其运行循环提供服务,并且不会调用 SIGINT 处理程序。由于SIGINT经常被用来正好中断这样一个卡住的进程,所以主线程可能不适合。
因此,您可能希望生成自己的线程来监控此信号。它不应该做任何其他事情,因为其他任何事情也可能导致它卡住。但是,即使在那里,也存在问题。您的处理函数将在您的此后台线程上调用,并且主线程可能仍被锁定。系统库中有很多仅主线程的东西,您将无法执行任何操作。但您将拥有比 POSIX 风格的信号处理程序更大的灵活性。
我应该补充一点,GCD 的调度源还可以监视 UNIX 信号并且更易于使用,尤其是在 Swift 中。但是,他们不会预先创建专用线程来运行处理程序。处理程序将被提交到队列。现在,您可以指定一个高优先级/高 QOS 队列,但我不完全确定如果该进程已经有大量失控线程正在运行,该处理程序是否会运行。也就是说,实际上在高优先级队列上运行的任务将优先于低优先级线程或队列,但启动新任务可能不会。我不确定。