请问为什么在WM_QUERYENDSESSION或者WM_ENDSESSION中发出WM_QUIT,消息循环无法终止?

沙老师 2006-09-19 11:18:28
请问为什么在WM_QUERYENDSESSION或者WM_ENDSESSION中发出WM_QUIT,消息循环无法终止?

#include <windows.h>
#include "EvntLog.h"
#include "Program.h"

// 类名
TCHAR szClassName[] = TEXT("ClassName");
// 标题
TCHAR szWindowName[] = TEXT("WindowName");

CEvntLog *pEvntLog;
CProgram *pProgram;

LRESULT CALLBACK WndProc(HWND, UINT, WPARAM, LPARAM);

int APIENTRY WinMain(HINSTANCE hInstance,
HINSTANCE hPrevInstance,
LPSTR lpCmdLine,
int nCmdShow)
{
MSG msg;
HWND hWnd;
WNDCLASS wc;

pEvntLog = new CEvntLog;
pProgram = new CProgram(pEvntLog);

// 注册窗口类
wc.style = CS_VREDRAW | CS_HREDRAW;
wc.lpfnWndProc = (WNDPROC)WndProc;
wc.cbClsExtra = 0;
wc.cbWndExtra = 0;
wc.hInstance = hInstance;
wc.hIcon = LoadIcon( NULL, IDI_APPLICATION );
wc.hCursor = LoadCursor( NULL, IDC_ARROW );
wc.hbrBackground = (HBRUSH)(COLOR_BTNFACE + 1);
wc.lpszMenuName = NULL;
wc.lpszClassName = szClassName;
RegisterClass(&wc);

// 创建窗口
hWnd = CreateWindow(szClassName, szWindowName, WS_OVERLAPPEDWINDOW,
CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT, CW_USEDEFAULT,
NULL, NULL, hInstance, NULL);

ShowWindow(hWnd, nCmdShow);
UpdateWindow(hWnd);

// 消息循环
while (GetMessage(&msg, NULL, 0, 0))
{
TranslateMessage(&msg);
DispatchMessage(&msg);
}

pEvntLog->WriteEvnt("Quitting...");

delete pProgram;
delete pEvntLog;

return msg.wParam;
}



LRESULT CALLBACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM lParam)
{

switch (message)
{
case WM_DESTROY:
pEvntLog->WriteEvnt("WM_DESTROY");
PostQuitMessage(0);
break;

case WM_QUERYENDSESSION:
if (lParam & ENDSESSION_LOGOFF) {
pEvntLog->WriteEvnt("LOGOFF");
}
else {
pEvntLog->WriteEvnt("SHUTDOWN");
}
// delete pProgram;
// delete pEvntLog;
DestroyWindow(hWnd);
return TRUE;


default:
return DefWindowProc(hWnd, message, wParam, lParam);
}

return 0;
}

程序在WinMain开头new了两个对象,本来指望在WinMain结尾,也就是消息循环结束后delete掉。但是这个delete过程在我正常关闭窗口时是可以执行的,注销(或关机)时候却无法正常执行。

注销(或关机)时候,程序先处理了WM_QUERYENDSESSION消息,执行了DestroyWindow,然后也正常处理了WM_DESTROY消息,这在程序的日志上可以看到,按理说PostQuitMessage后消息循环就应该终止了,然后程序正常delete对象,正常终止,但是事实上却没有delete。百思不得其解...
...全文
1300 18 打赏 收藏 转发到动态 举报
写回复
用AI写文章
18 条回复
切换为时间正序
请发表友善的回复…
发表回复
jazy 2006-09-22
  • 打赏
  • 举报
回复
因为你在
case WM_QUERYENDSESSION:
的最后
return TRUE;

WM_QUERYENDSESSION消息的处理一旦return,就关机了,要做什么操作,在return之前做
沙老师 2006-09-21
  • 打赏
  • 举报
回复
to jazy():

这里可以走的,但是为什么执行不了消息循环下面的语句呢?
jazy 2006-09-21
  • 打赏
  • 举报
回复
case WM_QUERYENDSESSION:
if (lParam & ENDSESSION_LOGOFF) {
pEvntLog->WriteEvnt("LOGOFF");
}
else {
pEvntLog->WriteEvnt("SHUTDOWN");
}
delete pProgram;
delete pEvntLog;
这里的delete走不到吗?不应该啊,return之前注销操作是在等待的
沙老师 2006-09-20
  • 打赏
  • 举报
回复
to jazy():

我的初衷是无论按关闭按钮还是注销,都能够停止消息循环进而执行循环下面的语句。但是现在发现注销时候不行,而且问题越来越奇怪。
jazy 2006-09-20
  • 打赏
  • 举报
回复
The application need not call the DestroyWindow or PostQuitMessage function when the session is ending.

你的代码为什么要注释那两个delete?原先不是合理的吗?
case WM_QUERYENDSESSION:
if (lParam & ENDSESSION_LOGOFF) {
pEvntLog->WriteEvnt("LOGOFF");
}
else {
pEvntLog->WriteEvnt("SHUTDOWN");
}
delete pProgram;
delete pEvntLog;
return TRUE;
沙老师 2006-09-20
  • 打赏
  • 举报
回复
case WM_QUERYENDSESSION:
if (lParam & ENDSESSION_LOGOFF) {
pEvntLog->WriteEvnt("LOGOFF");
}
else {
pEvntLog->WriteEvnt("SHUTDOWN");
}
// DestroyWindow(hWnd);
PostQuitMessage(0);
return TRUE;

试过了,还是不行。另外,DestroyWindow应该可以用在消息循环内部,它发出WM_DESTROY消息,然后在WM_DESTROY处理里调用PostQuitMessage发出WM_QUIT消息退出消息循环。
沙老师 2006-09-20
  • 打赏
  • 举报
回复
upupupup
沙老师 2006-09-19
  • 打赏
  • 举报
回复
正常关闭窗口和注销都能够执行到WM_DESTROY消息啊...
hanjk 2006-09-19
  • 打赏
  • 举报
回复
你在DestroyWindow(hWnd)之前不是已经delete了吗
yjgx007 2006-09-19
  • 打赏
  • 举报
回复
I think you are mistake to process window's default procedure.

Please move that code in 'switch' section as below to last line of WndProc(...)
======================================================
return DefWindowProc(hWnd, message, wParam, lParam);
Elysium 2006-09-19
  • 打赏
  • 举报
回复
只在本进程内有效
goodboyws 2006-09-19
  • 打赏
  • 举报
回复
好像不是这个问题,汗!
goodboyws 2006-09-19
  • 打赏
  • 举报
回复
呵呵,问题在这里,DestroyWindow是不可以用在消息循环内部的,直接用PostQuitMessage(0);就可以
case WM_QUERYENDSESSION:
if (lParam & ENDSESSION_LOGOFF) {
pEvntLog->WriteEvnt("LOGOFF");
}
else {
pEvntLog->WriteEvnt("SHUTDOWN");
}
//delete pProgram;
//delete pEvntLog;
PostQuitMessage(0);
return TRUE;
沙老师 2006-09-19
  • 打赏
  • 举报
回复
from MSDN
.....
When an application returns TRUE for this message, it receives the WM_ENDSESSION message and it is terminated, regardless of how the other applications respond to the WM_QUERYENDSESSION
因为你返回TRUE,系统会terminate你的进程,所以你的对象来不及释放
==========================================================
刚才试了,return FALSE 照样不行,系统可以注销,但是对象仍旧没有释放。如果将DestroyWindow改成CloseWindow,还是return FALSE ,那么系统不可以注销,窗口在注销的时候只是最小化而不关闭,呵呵。

goodboyws 2006-09-19
  • 打赏
  • 举报
回复
from MSDN
.....
When an application returns TRUE for this message, it receives the WM_ENDSESSION message and it is terminated, regardless of how the other applications respond to the WM_QUERYENDSESSION
因为你返回TRUE,系统会terminate你的进程,所以你的对象来不及释放
goodboyws 2006-09-19
  • 打赏
  • 举报
回复
在delete 之前已经关机了
沙老师 2006-09-19
  • 打赏
  • 举报
回复
解决问题的100分奉送!
沙老师 2006-09-19
  • 打赏
  • 举报
回复
高手怎么都这么吝啬阿。。。我只希望程序在注销时候能够正常退出(就是能够执行到“
pEvntLog->WriteEvnt("Quitting...");

delete pProgram;
delete pEvntLog;

return msg.wParam;
}

”)这个地方,不可能做到吗?
消息,就是指Windows发出的一个通知,告诉应用程序某个事情发生了。例如,单击鼠标、改变窗口尺寸、按下键盘上的一个键都会使Windows发送一个消息给应用程序。 消息本身是作为一个记录传递给应用程序的,这个记录包含了消息的类型以及其他信息。例如,对于单击鼠标所产生的消息来说,这个记录包含了单击鼠标时的坐标。这个记录类型叫做TMsg,它在Windows单元是这样声明的: type TMsg = packed record hwnd: HWND; //窗口句柄 message: UINT;//消息常量标识符 wParam: WPARAM ;// 32位消息的特定附加信息 lParam: LPARAM ;// 32位消息的特定附加信息 time: DWORD;//消息创建时的时间 pt: TPoint; //消息创建时的鼠标位置 end ; 消息有什么? 是否觉得一个消息记录的信息像希腊语一样?如果是这样,那么看一看下面的解释:hwnd 32位的窗口句柄。窗口可以是任何类型的屏幕对象,因为Win32能够维护大多数可 视对象的句柄(窗口、对话框、按钮、编辑框等)。message 用于区别其他消息的常量值,这些常量可以是Windows单元预定义的常量,也 可以是自定义的常量。 wParam 通常是一个与消息有关的常量值,也可能是窗口或控件的句柄。 lParam 通常是一个指向内存数据的指针。由于WParam、lParam和Pointer都是32位的,因此,它们之间可以相互转换。 WM_NULL =$0000 // WM_CREATE =$0001 //应用程序创建一个窗口 WM_DESTROY = $0002 //一个窗口被销毁 WM_MOVE = $0003 //移动一个窗口 WM_SIZE= $0005 //改变一个窗口的大小 WM_ACTIVATE= $0006 //一个窗口被激活或失去激活状态; WM_SETFOCUS= $0007 //获得焦点后 WM_KILLFOCUS= $0008 //失去焦点 WM_ENABLE= $000A //改变enable状态 WM_SETREDRAW= $000B //设置窗口是否能重画 WM_SETTEXT= $000C //应用程序发送此消息来设置一个窗口的文本 WM_GETTEXT = $000D //应用程序发送此消息来复制对应窗口的文本到缓冲区 WM_GETTEXTLENGTH = $000E //得到与一个窗口有关的文本的长度(不包含空字符) WM_PAINT = $000F //要求一个窗口重画自己 WM_CLOSE = $0010 //当一个窗口或应用程序要关闭时发送一个信号 WM_QUERYENDSESSION= $0011 //当用户选择结束对话框或程序自己调用ExitWindows函数 WM_QUIT= $0012 //用来结束程序运行或当程序调用postquitmessage函数 WM_QUERYOPEN = $0013 //当用户窗口恢复以前的大小位置时,把此消息发送给某个图标 WM_ERASEBKGND = $0014 //当窗口背景必须被擦除时(例在窗口改变大小时) WM_SYSCOLORCHANGE = $0015 //当系统颜色改变时,发送此消息给所有顶级窗口 WM_ENDSESSION = $0016 // 当系统进程发出WM_QUERYENDSESSION消息后,此消息发送给应用程序,通知它对话是否结束 WM_SYSTEMERROR = $0017 // WM_SHOWWINDOW= $0018 //当隐藏或显示窗口是发送此消息给这个窗口 WM_ACTIVATEAPP = $001C //发此消息给应用程序哪个窗口是激活的,哪个是非激活的; WM_FONTCHANGE= $001D //当系统的字体资源库变化时发送此消息给所有顶级窗口 WM_TIMECHANGE= $001E //当系统的时间变化时发送此消息给所有顶级窗口 WM_CANCELMODE= $001F //发送此消息来取消某种正在进行的摸态(操作) WM_SETCURSOR = $0020 //如果鼠标引起光标在某个窗口移动且鼠标输入没有被捕获时,就发消息给某个窗口 WM_MOUSEACTIVATE = $0021 //当光标在某个非激活的窗口而用户正按着鼠标的某个键发送此消息给当前窗口 WM_CHILDACTIVATE = $0022 //发送此消息给MDI子窗口当用户点击此窗口的标题栏,或当窗口被激活,移动,改变大小 WM_QUEUESYNC= $0023 //此消息由基
Windows消息大全易语言版, 消息,就是指Windows发出的一个通知,告诉应用程序某个事情发生了。例如,单击鼠标、改变窗口尺寸、按下键盘上的一个键都会使Windows发送一个消息给应用程序。消息本身是作为一个记录传递给应用程序的,这个记录包含了消息的类型以及其他信息。例如,对于单击鼠标所产生的消息来说,这个记录包含了单击鼠标时的坐标。这个记录类型叫做TMsg, 它在Windows单元是这样声明的: type TMsg = packed record hwnd: HWND; / /窗口句柄 message: UINT; / /消息常量标识符 wParam: WPARAM ; // 32位消息的特定附加信息 lParam: LPARAM ; // 32位消息的特定附加信息 time: DWORD; / /消息创建时的时间 pt: TPoint; / /消息创建时的鼠标位置 end; 消息有什么? 是否觉得一个消息记录的信息像希腊语一样?如果是这样,那么看一看下面的解释: hwnd 32位的窗口句柄。窗口可以是任何类型的屏幕对象,因为Win32能够维护大多数可视对象的句柄(窗口、对话框、按钮、编辑框等)。 message 用于区别其他消息的常量值,这些常量可以是Windows单元预定义的常量,也可以是自定义的常量。 wParam 通常是一个与消息有关的常量值,也可能是窗口或控件的句柄。 lParam 通常是一个指向内存数据的指针。由于W P a r a m、l P a r a m和P o i n t e r都是3 2位的, 因此,它们之间可以相互转换。 WM_NULL = 0; WM_CREATE = 1; 应用程序创建一个窗口 WM_DESTROY = 2; 一个窗口被销毁 WM_MOVE = 3; 移动一个窗口 WM_SIZE = 5; 改变一个窗口的大小 WM_ACTIVATE = 6; 一个窗口被激活或失去激活状态; WM_SETFOCUS = 7; 获得焦点后 WM_KILLFOCUS = 8; 失去焦点 WM_ENABLE = 10; 改变enable状态 WM_SETREDRAW = 11; 设置窗口是否能重画 WM_SETTEXT = 12; 应用程序发送此消息来设置一个窗口的文本 WM_GETTEXT = 13; 应用程序发送此消息来复制对应窗口的文本到缓冲区 WM_GETTEXTLENGTH = 14; 得到与一个窗口有关的文本的长度(不包含空字符) WM_PAINT = 15; 要求一个窗口重画自己 WM_CLOSE = 16; 当一个窗口或应用程序要关闭时发送一个信号 WM_QUERYENDSESSION = 17; 当用户选择结束对话框或程序自己调用ExitWindows函数 WM_QUIT = 18; 用来结束程序运行或当程序调用postquitmessage函数 WM_QUERYOPEN = 19; 当用户窗口恢复以前的大小位置时,把此消息发送给某个图标 WM_ERASEBKGND = 20; 当窗口背景必须被擦除时(例在窗口改变大小时) WM_SYSCOLORCHANGE = 21; 当系统颜色改变时,发送此消息给所有顶级窗口 WM_ENDSESSION = 22; 当系统进程发出WM_QUERYENDSESSION消息后,此消息发送给应用程序, 通知它对话是否结束 WM_SYSTEMERROR = 23; WM_SHOWWINDOW = 24; 当隐藏或显示窗口是发送此消息给这个窗口 WM_ACTIVATEAPP = 28; 发此消息给应用程序哪个窗口是激活的,哪个是非激活的; WM_FONTCHANGE = 29; 当系统的字体资源库变化时发送此消息给所有顶级窗口 WM_TIMECHANGE = 30; 当系统的时间变化时发送此消息给所有顶级窗口 WM_CANCELMODE = 31; 发送此消息来取消某种正在进行的摸态(操作) WM_SETCURSOR = 32; 如果鼠标引起光标在某个窗口移动且鼠标输入没有被捕获时,就发消息给某个窗口 WM_MOUSEACTIVATE = 33; 当光标在某个非激活的窗口而用户正按着鼠标的某个键发送此消息给当前窗口 WM_CHILDACTIVATE = 34; 发送此消息给MDI子窗口当用户点击此窗口的标题栏,或当窗口被激活,移动,改变大小 WM_QUEUESYNC = 35; 此消息由基于计算机的训练程序发送,通过WH_JOURNALPALYBACK的hook程序 分离出用户输入消息 WM_GETMINMAXINFO = 36; 此消息发送给窗口当它将要改变大小或位置; WM_PAINTICON = 38; 发送给最小化窗口当它图标将

16,472

社区成员

发帖
与我相关
我的任务
社区描述
VC/MFC相关问题讨论
社区管理员
  • 基础类社区
  • Web++
  • encoderlee
加入社区
  • 近7日
  • 近30日
  • 至今
社区公告

        VC/MFC社区版块或许是CSDN最“古老”的版块了,记忆之中,与CSDN的年龄几乎差不多。随着时间的推移,MFC技术渐渐的偏离了开发主流,若干年之后的今天,当我们面对着微软的这个经典之笔,内心充满着敬意,那些曾经的记忆,可以说代表着二十年前曾经的辉煌……
        向经典致敬,或许是老一代程序员内心里面难以释怀的感受。互联网大行其道的今天,我们期待着MFC技术能够恢复其曾经的辉煌,或许这个期待会永远成为一种“梦想”,或许一切皆有可能……
        我们希望这个版块可以很好的适配Web时代,期待更好的互联网技术能够使得MFC技术框架得以重现活力,……

试试用AI创作助手写篇文章吧