Mendaftarkan Fungsi Handler Kontrol
Contoh Handler Kontrol Dasar
Ini adalah contoh fungsi SetConsoleCtrlHandler yang digunakan untuk menginstal handler kontrol.
Ketika sinyal CTRL+C diterima, handler kontrol mengembalikan TRUE, yang menunjukkan bahwa ia telah menangani sinyal. Melakukan ini mencegah penangan kontrol lain dipanggil.
Saat sinyal CTRL_CLOSE_EVENT diterima, handler kontrol mengembalikan TRUE dan proses berakhir.
Saat sinyal CTRL_BREAK_EVENT, CTRL_LOGOFF_EVENT, atau CTRL_SHUTDOWN_EVENT diterima, handler kontrol mengembalikan FALSE. Melakukan ini menyebabkan sinyal diteruskan ke fungsi handler kontrol berikutnya. Jika tidak ada handler kontrol lain yang telah terdaftar atau tidak ada handler terdaftar yang mengembalikan TRUE, handler default akan digunakan, yang mengakibatkan proses dihentikan.
Catatan
Memanggil AttachConsole, AllocConsole, atau FreeConsole akan mengatur ulang tabel penangan kontrol dalam proses klien ke status awalnya. Handler harus didaftarkan lagi ketika sesi konsol yang terpasang berubah.
// CtrlHandler.cpp : This file contains the 'main' function. Program execution begins and ends there.
//
#include <windows.h>
#include <stdio.h>
BOOL WINAPI CtrlHandler(DWORD fdwCtrlType)
{
switch (fdwCtrlType)
{
// Handle the CTRL-C signal.
case CTRL_C_EVENT:
printf("Ctrl-C event\n\n");
Beep(750, 300);
return TRUE;
// CTRL-CLOSE: confirm that the user wants to exit.
case CTRL_CLOSE_EVENT:
Beep(600, 200);
printf("Ctrl-Close event\n\n");
return TRUE;
// Pass other signals to the next handler.
case CTRL_BREAK_EVENT:
Beep(900, 200);
printf("Ctrl-Break event\n\n");
return FALSE;
case CTRL_LOGOFF_EVENT:
Beep(1000, 200);
printf("Ctrl-Logoff event\n\n");
return FALSE;
case CTRL_SHUTDOWN_EVENT:
Beep(750, 500);
printf("Ctrl-Shutdown event\n\n");
return FALSE;
default:
return FALSE;
}
}
int main(void)
{
if (SetConsoleCtrlHandler(CtrlHandler, TRUE))
{
printf("\nThe Control Handler is installed.\n");
printf("\n -- Now try pressing Ctrl+C or Ctrl+Break, or");
printf("\n try logging off or closing the console...\n");
printf("\n(...waiting in a loop for events...)\n\n");
while (1) {}
}
else
{
printf("\nERROR: Could not set control handler");
return 1;
}
return 0;
}
Dengarkan dengan Contoh Jendela Tersembunyi
Per keterangan, jika pustaka gdi32.dll atau user32.dll dimuat, SetConsoleCtrlHandler tidak dipanggil untuk peristiwa CTRL_LOGOFF_EVENT dan CTRL_SHUTDOWN_EVENT . Solusi yang ditentukan adalah membuat jendela tersembunyi, jika tidak ada jendela yang sudah ada, dengan memanggil metode CreateWindowEx dengan parameter dwExStyle yang diatur ke 0 dan mendengarkan pesan jendela WM_QUERYENDSESSION dan WM_ENDSESSION. Jika jendela sudah ada, tambahkan dua pesan ke Prosedur Jendela yang ada.
Informasi selengkapnya tentang menyiapkan jendela dan perulangan olahpesannya dapat ditemukan di Membuat Jendela.
// CtrlHandler.cpp : This file contains the 'main' function. Program execution begins and ends there.
//
#include <Windows.h>
#include <stdio.h>
BOOL WINAPI CtrlHandler(DWORD fdwCtrlType)
{
switch (fdwCtrlType)
{
// Handle the CTRL-C signal.
case CTRL_C_EVENT:
printf("Ctrl-C event\n\n");
Beep(750, 300);
return TRUE;
// CTRL-CLOSE: confirm that the user wants to exit.
case CTRL_CLOSE_EVENT:
Beep(600, 200);
printf("Ctrl-Close event\n\n");
return TRUE;
// Pass other signals to the next handler.
case CTRL_BREAK_EVENT:
Beep(900, 200);
printf("Ctrl-Break event\n\n");
return FALSE;
case CTRL_LOGOFF_EVENT:
Beep(1000, 200);
printf("Ctrl-Logoff event\n\n");
return FALSE;
case CTRL_SHUTDOWN_EVENT:
Beep(750, 500);
printf("Ctrl-Shutdown event\n\n");
return FALSE;
default:
return FALSE;
}
}
LRESULT CALLBACK WindowProc(HWND hwnd, UINT uMsg, WPARAM wParam, LPARAM lParam)
{
switch (uMsg)
{
case WM_QUERYENDSESSION:
{
// Check `lParam` for which system shutdown function and handle events.
// See https://learn.microsoft.com/windows/win32/shutdown/wm-queryendsession
return TRUE; // Respect user's intent and allow shutdown.
}
case WM_ENDSESSION:
{
// Check `lParam` for which system shutdown function and handle events.
// See https://learn.microsoft.com/windows/win32/shutdown/wm-endsession
return 0; // We have handled this message.
}
default:
return DefWindowProc(hwnd, uMsg, wParam, lParam);
}
}
int main(void)
{
WNDCLASS sampleClass{ 0 };
sampleClass.lpszClassName = TEXT("CtrlHandlerSampleClass");
sampleClass.lpfnWndProc = WindowProc;
if (!RegisterClass(&sampleClass))
{
printf("\nERROR: Could not register window class");
return 2;
}
HWND hwnd = CreateWindowEx(
0,
sampleClass.lpszClassName,
TEXT("Console Control Handler Sample"),
0,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
CW_USEDEFAULT,
nullptr,
nullptr,
nullptr,
nullptr
);
if (!hwnd)
{
printf("\nERROR: Could not create window");
return 3;
}
ShowWindow(hwnd, SW_HIDE);
if (SetConsoleCtrlHandler(CtrlHandler, TRUE))
{
printf("\nThe Control Handler is installed.\n");
printf("\n -- Now try pressing Ctrl+C or Ctrl+Break, or");
printf("\n try logging off or closing the console...\n");
printf("\n(...waiting in a loop for events...)\n\n");
// Pump message loop for the window we created.
MSG msg{};
while (GetMessage(&msg, nullptr, 0, 0) > 0)
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}
return 0;
}
else
{
printf("\nERROR: Could not set control handler");
return 1;
}
}