2017-12-11 3 views
-2

사용자가 을 누르면 콘솔 창 오른쪽 상단에있는 작은 "x"를 눌러 정리 기능을 호출하고 싶습니다.콘솔에서 트랩 종료

atexit 메서드를 등록했지만이 경우 호출되지 않습니다.

해결책은 Windows 및 Linux에서 작동해야합니다.

+1

언어/플랫폼이란 무엇입니까? 이 이벤트에서 어떤 앱을 호출해야합니까? 콘솔에서 실행중인 앱이 알림을 받아야하거나 외부 앱과 같은 것이어야합니까? –

+1

'콘솔 종료 트랩'... 괜찮아 ... '플랫폼 독립적 인 '나는 듣기 '작은 x'를 클릭하면 _ 플랫폼 독립적 인 small x을 의미합니까? –

답변

1

여기에서는 atexit을 사용할 수 없습니다. 프로세스가 정상적으로 종료 될 때 호출되기 때문입니다. 반면에 프로세스가 신호로 종료되는 경우가 있기 때문입니다.

linux에서 제어 터미널이 닫히면 SIGHUP (신호 중단)이 프로세스로 보내지고 POSIX 신호 처리를 사용할 수 있습니다.

windows에서 CTRL_CLOSE_EVENT 이벤트가 전달되고 SetConsoleCtrlHandler winapi를 사용하여 처리 할 수 ​​있습니다.

제 지식에 따라 c++에는 플랫폼 독립적 인 방법이 없습니다. 다음 간단한 프로그램에 표시된 것처럼 플랫폼 종속 코드를 사용해야합니다. 나는 윈도우즈에서 VS와 우분투에서 g ++를 테스트했다. 오류 처리가 생략되었고 단순화를 위해 신호 처리기에서 I/O가 수행됩니다.

이 프로그램은 신호 처리기와 atexit 기능을 등록합니다. 그런 다음 10 초 동안 잔다. 10 초 이내에 콘솔을 닫지 않으면 프로세스가 정상적으로 종료되고 atexit 핸들러가 호출됩니다. 10 초 전에 창을 닫으면 신호가 울립니다. 리눅스에서는 시그널 핸들러가 호출되는 것을 보지 못할 것이다.하지만 호출되는 것은 아니다. 필자는 파일을 작성하거나 비프 음을 발생시켜 이것을 확인할 수있다.

#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 년 동안 가지고 있었던 문제를 해결합니다. \t CTRL_BREAK_EVENT : 및 \t CTRL_SHUTDOWN_EVENT : 나는 \t CTRL_C_EVENT에 대한 처리기를 추가하는 데 유용 것을 발견했다 : 창문 핸들러 신호 것이 : – Jacko

+0

@Jacko 다행. 이것들은 내가 다루고 싶은 사건이기도합니다. – dhanushka