欢迎您访问 最编程 本站为您分享编程语言代码,编程技术文章!
您现在的位置是: 首页

快速退出

最编程 2024-01-15 17:46:53
...

快速退出:quick_exit 与 at_quick_exit

一些程序终止函数: terminate, abort, exit

  • 正常退出
  • 异常退出

terminate

terminate 函数实际上是 C++ 语言中异常处理的一部分(<exception>).
一般而言,没有被捕捉的异常就会导致 terminate 函数的调用。

noexcept 关键字声明的函数,如果抛出了异常,也会调用 terminate 函数。

只要 C++ 程序中出现了非程序员预期的行为,都有可能导致 terminate 函数的调用。

terminate 函数默认情况下,会调用 abort 函数。

用户可以通过 set_terminate 函数来改变默认行为。

abort

  • <cstdlib>
  • abort 函数不会调用任何析构函数
  • 默认情况下,abort 会向系统抛出一个信号(signal):SIGABRT。如果程序员为信号设定一个信号处理函数(signal handler)的话,那么系统将默认地释放进程的所有资源,从而终止程序。
  • abort 是 终止进程

exit

  • 正常退出
  • exit 会正常调用自动变量的析构函数,并且还会调用 atexit 注册的函数
#include <cstdlib>
#include <iostream>

void openDevice()
{
    std::cout << "device is opened." << std::endl;
}

void resetDeviceState()
{
    std::cout << "device stat is reset." << std::endl;
}

void closeDevice()
{
    std::cout << "device is closed." << std::endl;
}

int main()
{
    atexit(closeDevice);
    atexit(resetDeviceState);
    openDevice();
    exit(0);

    return 0;
}
/*
device is opened.
device stat is reset.
device is closed.
*/

exit 的问题?
main或者其他使用 exit() 结束的程序可能在堆空间上有大量零散的内存,而main或者 exit 函数调用会导致类的析构函数依次将这些零散的内存释放。这是一件耗时的工作,而实际上,这些堆内存将在进程结束的时候由操作系统统一回收(事实上,这非常快,操作系统除了释放一些进程相关的数据结构外,只是将一些物理内存标记为未使用即可)。

如果堆内存对其他程序不产生影响,那么程序结束时释放堆内存的析构过程毫无意义。因此,我们常常需要能够更快速的退出程序。

另外多线程的情况下,使用 exit 来退出的话,通常需要向线程发出一个信号,并等待线程结束后再执行析构函数、atexit注册的函数等。

在C++11中引入了 quick_exit 函数,该函数并不执行析构函数而只是使进程终止。与 abort 不同的是,abort 结果通常是异常退出(可能会进行coredump),而 quick_exit 和 exit 属于正常退出。

使用 at_quick_exit 注册的函数也可以在 quick_exit 的时候被调用。这样以来,我们同样可以像 exit 一样做一些清理工作。

在 C++11 中,at_quick_exit 和 at_exit 一样,标准要求编译器至少支持32个注册函数的调用。

#include <cstdlib>
#include <iostream>

struct A {
    ~A() {
        std::cout << "Destruct A." << std::endl;
    }
};

void closeDevice() {
    std::cout << "devie is closed." << std::endl;
}

int main()
{
    A a;
    at_quick_exit(closeDevice);
    quick_exit(0);
}
// devie is closed.