【问题标题】:Trap exit on console控制台上的陷阱退出
【发布时间】:2017-12-11 14:38:39
【问题描述】:

我想在用户按下后调用清理函数 控制台窗口右上角的小“x”。

我已经注册了一个atexit 方法,但在这种情况下它不会被调用。

解决方案需要在 windows 和 linux 上运行。

【问题讨论】:

  • 语言/平台是什么?在这个事件中应该调用什么应用程序?是应该通知在控制台中运行的应用程序,还是应该通知一些外部应用程序?
  • console exit trap ... 好吧 ... platform independant 我在听click the small x 你是说平台无关 小x吗?

标签: c++ linux windows console atexit


【解决方案1】:

你不能在这里使用atexit,因为它是在进程正常终止时调用的,而在你的情况下,进程是由信号终止的。

linux,SIGHUP(信号挂断)在其控制终端关闭时发送给进程,您可以使用POSIX信号处理。

windows,传递CTRL_CLOSE_EVENT事件,你可以使用SetConsoleCtrlHandlerwinapi来处理。

因此,据我所知,c++ 中没有独立于平台的方法来处理此问题。您应该使用平台相关代码来执行此操作,如以下简单程序所示。我在带有 VS 的 windows 和带有 g++ 的 ubuntu 上对其进行了测试。请注意,为简单起见,错误处理已被省略,I/O 是在信号处理程序中执行的。

程序注册一个信号处理程序和一个atexit 函数。然后它会休眠 10 秒。如果您在 10 秒内没有关闭控制台,进程将正常终止并调用 atexit 处理程序。如果您在 10 秒前关闭窗口,它会捕捉到信号。在 linux 上,您不会看到信号处理程序被调用,但它确实被调用了,您可以通过写入文件(或发出哔声?)来检查这一点,尽管我不推荐这样做。

#ifdef  WIN32
#include <windows.h> 
#else
#include <unistd.h>
#include <signal.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#endif

#include <stdio.h>
#include <stdlib.h>


#ifdef  WIN32
BOOL sig_handler(DWORD signum) 
{ 
    switch( signum ) 
    { 
        case CTRL_CLOSE_EVENT: 
            printf( "Ctrl-Close event\n" );
            return( TRUE ); 

        default: 
            return FALSE; 
    } 
}
#else
void sig_handler(int signum)
{
    /* you won't see this printed, but it runs */
    printf("Received signal %d\n", signum);
}
#endif 


void exit_fn(void)
{
   printf("%s\n", __FUNCTION__);
}


void setup_signal_handler()
{
#ifdef  WIN32
    SetConsoleCtrlHandler((PHANDLER_ROUTINE)sig_handler, TRUE);
#else
    struct sigaction sa;
    sa.sa_handler = &sig_handler;
    sigfillset(&sa.sa_mask);
    sigaction(SIGHUP, &sa, NULL);
#endif  
}


int main(void)
{
    printf("%s\n", __FUNCTION__);
    /* setup signal handler */
    setup_signal_handler();
    /* setup function to be called at normal process termination */
    atexit(exit_fn);
    /* sleep for 10s */
#ifdef  WIN32
    Sleep(10000);
#else
    sleep(10);
#endif
    /* print message if process terminates normally */
    printf("Normal process termination\n");

    exit(EXIT_SUCCESS);
}

【讨论】:

  • 太棒了!非常感谢。
  • 像魅力一样工作。再次感谢你!这解决了我至少一年以来一直存在的问题。
  • @Jacko 很高兴它有帮助:)
  • 一个附录:我发现为 CTRL_C_EVENT: CTRL_BREAK_EVENT: 和 CTRL_SHUTDOWN_EVENT: 添加处理程序对 windows 信号处理程序很有用。因为这些也是我想处理的事件。
猜你喜欢
  • 2020-06-22
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-06-25
  • 2019-05-15
  • 1970-01-01
  • 1970-01-01
  • 2013-06-02
相关资源
最近更新 更多