3

我正在尝试使用类中的成员函数进行中断处理。

代码是

signal(SIGABRT, socketServer.signalHandler);

其中的定义signalHandler

public:
void SocketServer::signalHandler(int sig)
{
  logger.info("Receive SIG"+to_string(sig)+" Stopping server now...");
  stop();
}

当我编译代码时,我得到一个错误说

main.cpp:32:32: error: reference to non-static member function must be called
  signal(SIGABRT, socketServer.signalHandler);

我正在尝试SIGABRT使用此signalHandler功能进行捕获并清理和停止socketServer实例。我想我可以使用全局变量和全局函数来完成这项工作,但是关于使用成员函数这样做有什么想法吗?

4

1 回答 1

3

不,你不能这样做。

原因是所有成员函数都有一个“隐含/隐藏”的this指针参数。如果我们“扁平化”您的处理程序定义以生成 C 等效项,它将如下所示:

void SocketServer::signalHandler(SocketServer *this,int sig);

[在 C 中]的signal函数对这个 [双关语] 一无所知。如果它已编译,则将调用处理程序并sig进入this参数而不是参数sig

所以,你真的必须这样做:

SocketServer my_global_server;

void
my_handler(int sig)
{

    my_global_server.signalHandler(sig);
}

int
main(void)
{

    signal(SIGABRT,my_handler);

    return 0;
}

实际上,上述情况是相当危险的,因为my_global_server在调用信号处理程序时可能处于不确定状态,从而导致 UB。此外,在信号处理程序中,您可以做的事情数量有限。例如,不允许进行堆操作。

这是实现此功能的更好方法:

volatile int signal_flag;

SocketServer my_global_server;

void
my_handler(int sig)
{

    signal_flag = sig;
}

int
main(void)
{

    signal(SIGABRT,my_handler);

    while (! signal_flag) {
        ...
    }

    my_global_server.signalHandler(signal_flag);

    return 0;
}
于 2015-12-15T21:51:34.210 回答