我正在寻找一种方法来停止每2秒执行一次任务的线程。我决定尝试使用std::promise/future,以便在设置promise时线程可以立即退出。
#include <future>
#include <iostream>
#include <chrono>
#include <csignal>
std::promise<void> stop;
int main() {
std::signal(SIGINT, [] (int) { stop.set_value(); } );
auto future = stop.get_future();
while (future.wait_for(std::chrono::seconds(1)) != std::future_status::ready) {
std::cout << "I'm still there" << std::endl;
}
}
事实上,这不起作用,并以这种方式崩溃:
$/a.out我还在那里^Cterminate在抛出"std::system_error"的实例后调用
what((:未知错误-1放弃(堆芯倾倒(
好的,一个人应该注意他在处理程序上下文中所做的事情,但我必须说,我没想到会发生这种崩溃;我真的不明白。。。你知道吗?
我相信这里的行为是未定义的,这是在信号处理程序中调用标准库函数(不在信号安全函数列表中(的结果。
信号处理程序函数(用户定义(的限制非常广泛,本文对此进行了说明。
另一件需要注意的事情是,如果信号处理程序引用的任何对象的静态或线程本地(自C++11以来(存储持续时间不是std::atomic
(自C++11以来(或volatile std::sig_atomic_t
。
正如@BasileStarynkevitch所指出的,如果您在Linux上,请确保从信号处理程序中只调用异步信号安全函数。