如何在SIGSEGV之后销毁在main()中创建的对象

时间:2014-09-04 11:36:38

标签: c++ signals exit resource-cleanup

我想知道在应用程序刚崩溃(或被中断,或其他一些)之后,是否存在任何破坏main()中创建的对象的琐碎方法。请参阅代码:

#include <iostream>
#include <signal.h>
#include <stdlib.h>

class TestClass {
public:
    TestClass() { std::cerr << "Constructor" << std::endl; }
    ~TestClass() { std::cerr << "Destructor" << std::endl; }
};

void signal_handler(int signum) {
    std::cerr << "Signal caught " << signum << std::endl;
    exit(1);
}

int main() {

    TestClass a; 

    struct sigaction new_action, old_action;
    new_action.sa_handler = signal_handler;
    sigemptyset (&new_action.sa_mask);
    new_action.sa_flags = 0;

    sigaction (SIGSEGV, NULL, &old_action);
    if (old_action.sa_handler != SIG_IGN)
    {
        sigaction (SIGSEGV, &new_action, NULL);
    }
    // CRASH !!!
    int* p = 0; *p = 0;
}

所以,我安装了一个信号处理程序,在执行时应该进行一些清理,然后退出。显然,它是一个很大的应用程序,所以我想调用我的类的析构函数来创建临时文件(从理论上讲)。

动态分配它并在signal_handler中释放它(为了调用析构函数)很容易,但这需要一些额外的工作。

而且,它可能是一个全局对象,但我想避免全局初始化顺序问题......

1 个答案:

答案 0 :(得分:1)

您可以尝试在SIGSEGV上协调有序关闭。这不是没有陷阱,而是从这里开始:Is it possible to terminate only the one thread on receiving a SIGSEGV?

您可以尝试终止信号处理程序中的一个违规线程,并设置一个全局标志,说“我们搞砸了,时间关闭”。然后在主消息循环中,检查标志并确定是否已设置。