首页 > 解决方案 > 如何在 C++ 中使用 std::bind 函数作为信号处理程序?

问题描述

我正在使用以下代码向我的 C++ 类添加信号处理:

namespace {
    std::atomic<bool> signal_flag(false);   
}
void terminate_or_interrupt_handler(int signal) {
    switch (signal) {
        case SIGTERM:
            WARN("SIGTERM received");
            signal_flag.store(true);
            break;
        case SIGINT:
            WARN("SIGINT received");
            signal_flag.store(true);
            break;
        default:
            throw (std::runtime_error("Unhandled signal received"));
    }
}
signal(SIGTERM, &terminate_or_interrupt_handler);

此代码有效,但它需要在与信号标志变量相同的范围内定义信号处理函数。我决定修改代码并通过signal_flag引用传递函数并使用std::bind将处理程序“专门化”到我的类。

 void terminate_or_interrupt_handler(std::atomic<bool>& signal_flag, int signal) {
    switch (signal) {
        case SIGTERM:
            WARN("SIGTERM received");
            signal_flag.store(true);
            break;
        case SIGINT:
            WARN("SIGINT received");
            signal_flag.store(true);
            break;
        default:
            throw (std::runtime_error("Unhandled signal received"));
    }
}
auto my_handler = std::bind(terminate_or_interrupt_handler, std::ref(my_class_signal_flag), std::placeholders::_1);
signal(SIGTERM, &my_handler);

但是,我得到这个编译错误:

error: cannot convert ‘std::_Bind<void (*(std::reference_wrapper<std::atomic<bool> >, std::_Placeholder<1>))(std::atomic<bool>&, int)>*’ to ‘__sighandler_t’ {aka ‘void (*)(int)’}

有没有办法将绑定函数与signalC++ 中的函数结合使用?

标签: c++signalsstdbind

解决方案


的结果std::bind是一个未指定的函数对象,其类型无法转换为void (*)(int). 尝试封装它:

void handler_foo(int signal)
{
    return terminate_or_interrupt_handler(signal_flag, signal);
}

或者,如果 C++11 可用,则 lambda 可能会更好:

signal(SIGTERM, [](int signal) { return terminate_or_interrupt_handler(signal_flag, signal); });

请注意,由于signal_flag是全局变量(命名空间范围变量),因此不需要捕获。非捕获 lambda 可以隐式转换为相应的函数指针类型。


推荐阅读