绑定静态成员函数作为回调

时间:2017-11-20 12:09:51

标签: c++ c++11 bind

我尝试使用std::bind将成员函数用作sigaction标头中termios.h的回调。我理解成员函数需要特殊照顾,并阅读并遵循示例herehere,没有这样的运气。

因为我可以通过静态函数,我想如果我把函数保留为s​​tatic,添加第二个变量作为指向自身的指针(this),我会很好,但没有这样的运气:

// In SerialListener.h
static void callback(int status, SerialListener *ptr);

// In the serial listener constructor

// Set callback
auto cb = std::bind(&SerialListener::callback, std::placeholders::_1, this);
sigAct_.sa_handler = cb;

但错误如下:

error: cannot convert ‘std::_Bind<void (*std::_Placeholder<1>, SerialListener*))(int, SerialListener*)>’ to ‘__sighandler_t {aka void (*)(int)}’ in assignment sigAct_.sa_handler = cb;

我还尝试了第二个例子中没有静态实现的变体:

// In SerialListener.h
void callback(int status);

// In the serial listener constructor

// Set callback
auto cb = std::bind(&SerialListener::callback, this, std::placeholders::_1);
sigAct_.sa_handler = cb;

哪个产生......

error: cannot convert ‘std::_Bind<std::_Mem_fn<void (SerialListener::*)(int)>(SerialListener*, std::_Placeholder<1>)>’ to ‘__sighandler_t {aka void (*)(int)}’ in assignment sigAct_.sa_handler = cb;

错误看起来非常相似,看起来它不能隐式地将绑定转换为它需要的处理程序,但是当我从输入类型和返回类型的角度来看它时它应该工作。我错过了什么?

1 个答案:

答案 0 :(得分:2)

lambda或bind都不能帮助您设置一个与信号处理程序签名不匹配的函数作为信号处理程序。请注意,信号处理函数签名不提供任何通过不透明指针传递类实例的方法。您可以做的最好的事情是在一些静态变量上存储指向处理上下文对象的指针:

SerialListener * p_listener;

void On_Signal(int signal_code)
{
    if(p_listener)
    {
         p_listener->DealWith(signal_code); // calling member function
    }
}

p_listener= &my_listener;

sigAct_.sa_handler = &On_Signal;

另请注意,根据信号的性质,信号处理程序中允许的操作范围可能非常有限。