3

我想捕捉 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);

预期的机制就是这样WindowProcPostQuitMessage,导致主消息循环接收WM_QUIT、中断循环并将程序发送到清理。当我退出程序(因此发送WM_DESTROY)时,程序会创建success.txt,但是当程序运行并且我注销(发送WM_ENDSESSION)时,它不会。

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

4

2 回答 2

3

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

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

于 2012-10-21T23:36:21.683 回答
0

在 Windows UI 应用程序中,您可以使用: 您可以从此msdn 链接 获得更多帮助。LRESULT CMainDlg::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{ if(message == WM_ENDSESSION) { if(lParam == ENDSESSION_LOGOFF) { /*Handle event*/ } } return CDialogEx::WindowProc(message, wParam, lParam); }

于 2014-06-10T12:28:51.473 回答