2017-12-11 11 views
-2

ユーザーが を押すと、コンソールウィンドウの右上隅に小さい「x」が表示された後で、クリーンアップ機能を呼びたいと思います。コンソールのトラップ出口

私はatexitメソッドを登録しましたが、この場合は呼び出されません。

ソリューションは、WindowsとLinuxで動作する必要があります。

+1

言語/プラットフォームとは何ですか?このイベントではどのアプリを呼び出す必要がありますか?それはコンソールで実行されているアプリのように、通知する必要がありますか、またはいくつかの外部アプリですか? –

+1

'コンソール終了トラップ' ...申し分なく... 'プラットフォームは独立しています。私はリッスンします '小さなxをクリックしてください 'あなたは_プラットフォームの独立した小さなxを意味しますか? –

答えて

1

ここではatexitを使用することはできません。これは、プロセスが正常に終了したときに呼び出されるためですが、プロセスではシグナルによって終了されるからです。

linuxで、制御端末が閉じられているときにプロセスにSIGHUP(シグナルハングアップ)が送信され、POSIX信号処理を使用できます。

windowsでCTRL_CLOSE_EVENTイベントが配信され、SetConsoleCtrlHandler winapiを使用してこれを処理できます。

私の知る限り、c++でこれを処理するプラットフォームに依存しない方法はありません。次の簡単なプログラムに示すように、プラットフォーム依存のコードを使用してください。私はVSでウィンドウズと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); 
} 
+0

優秀!どうもありがとうございました。 – Jacko

+0

魅力のように動作します。ありがとうございました!これは私が今少なくとも1年間持っていた問題を解決します。 – Jacko

+0

それは:) – dhanushka

関連する問題