我想捕获一个 Windows 注销事件,以便我可以进行一些清理。我的 WindowProc 如下所示:

switch (uMsg){
case WM_ENDSESSION:
case WM_DESTROY:
    PostQuitMessage(0);
    return 0;
// other messages
}
return DefWindowProc(hwnd, uMsg, wParam, lParam);
WinMain 中的消息循环如下所示:
for(;;){
    bool bTerminate = false;
    while(PeekMessage(&msg, NULL, 0, 0, PM_REMOVE)){
        if(msg.message == WM_QUIT){
            bTerminate = true;
        }
        TranslateMessage(&msg);
        DispatchMessage(&msg);
    }
    if(bTerminate){
        break;
    }
    // do other stuff
    Sleep(10);
}
FILE * fout;
fopen_s(&fout, "C:\\success.txt", "w"); // simulating cleanup actions
fclose(fout);
ExitProcess(0);

预期的机制是 WindowProc 执行 PostQuitMessage ,导致主消息循环接收 WM_QUIT ,打破循环并将程序发送到清理。当我退出程序(因此发送 WM_DESTROY )时,程序会创建 success.txt ,但是当程序正在运行并且我注销(发送 WM_ENDSESSION )时,它不会。

我也看过 WM_QUERYENDSESSION,但 MSDN 说“每个应用程序在收到此消息后应立即返回 TRUEFALSE,并推迟任何清理操作,直到它收到 WM_ENDSESSION 消息。”

最佳答案

WM_ENDSESSION 处理实际上并没有给您的应用程序退出消息循环的机会。您应该假设系统在发送 WM_ENDSESSION 消息后调用 TerminateProcess。

因此,您的应用程序需要执行的任何清理工作都应该在从窗口过程返回之前完成。

关于C++:响应 Windows 注销的清理操作,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/13003165/

10-14 12:35