鱼C论坛

 找回密码
 立即注册
查看: 63|回复: 1

[已解决]关于DispatchMessage函数的疑问

[复制链接]
发表于 2025-5-4 12:05:42 | 显示全部楼层 |阅读模式

马上注册,结交更多好友,享用更多功能^_^

您需要 登录 才可以下载或查看,没有账号?立即注册

x
#include <windows.h>

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

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)
{
        static TCHAR szAppName[] = TEXT("MyWindows");
        HWND hwnd;
        MSG msg;
        WNDCLASS wndclass;
       
        wndclass.style = CS_HREDRAW | CS_VREDRAW;
        wndclass.lpfnWndProc = WndProc;
        wndclass.cbClsExtra = 0;
        wndclass.cbWndExtra = 0;
        wndclass.hInstance = hInstance;
        wndclass.hIcon = LoadIcon(NULL, IDI_APPLICATION);
        wndclass.hCursor = LoadCursor(NULL, IDC_ARROW);
        wndclass.hbrBackground = (HBRUSH)GetStockObject(WHITE_BRUSH);
        wndclass.lpszMenuName = NULL;
        wndclass.lpszClassName = szAppName;
       
        if(!RegisterClass(&wndclass))
        {
                MessageBox(NULL, TEXT("这个程序需要在 Windows NT 才能执行!"), szAppName, MB_ICONERROR);
                return 0;
        }
       
        hwnd = CreateWindow(szAppName,
                        TEXT("鱼C工作室"),
                        WS_OVERLAPPEDWINDOW,
                        CW_USEDEFAULT,
                        CW_USEDEFAULT,
                        CW_USEDEFAULT,
                        CW_USEDEFAULT,
                        NULL,
                        NULL,
                        hInstance,
                        NULL);
                       
                ShowWindow(hwnd, iCmdShow);
                UpdateWindow(hwnd);
               
                while (GetMessage(&msg, NULL, 0, 0))
                {
                        TranslateMessage(&msg);
                        //DispatchMessage(&msg);
                        WndProc(msg.hwnd, msg.message, msg.wParam, msg.lParam);/*==============注意这一行=============================*/
                }
               
        return msg.wParam;
}
LRESULT CALLBACK WndProc(HWND hwnd, UINT message, WPARAM wParam, LPARAM lParam)
{
        HDC hdc;
        PAINTSTRUCT ps;
        RECT rect;
       
        wchar_t ch;
       
        switch (message)
        {
                case WM_PAINT:
                        hdc = BeginPaint(hwnd, &ps);
                        GetClientRect(hwnd, &rect);
                       
                        TextOut (hdc, 300, 300, TEXT("大家好,这是我的第一个窗口程序!"), sizeof(TEXT("大家好,这是我的第一个窗口程序!"))-1);
                        EndPaint(hwnd, &ps);
                        return 0;
               
                case WM_DESTROY:
                        PostQuitMessage(0);
                        return 0;
                       
                case WM_CHAR:
                        ch = wParam;
                        if(ch == (wchar_t)'a')
                        MessageBox(hwnd, TEXT("你好!"), TEXT("打招呼"), MB_OKCANCEL);
                       
        }
       
        return DefWindowProc(hwnd, message, wParam, lParam);
}

DispatchMessage是回调我们写的WndProc函数来处理消息,那我们能不能直接调用WndProc函数,将GetMessage接收的消息传给WndProc(像上面那样)
最佳答案
2025-5-4 12:06:21
问题:为什么不能直接调用WndProc代替DispatchMessage?


关键问题在于窗口过程的调用规范和时间戳处理!

虽然直接调用WndProc看似可行,但会引发以下问题:

  • 1. 缺少消息预处理
    DispatchMessage会在调用窗口过程前:
    - 设置线程的"上次消息时间戳"(GetMessageTime依赖这个)
    - 处理ANSI/Unicode转换
    - 处理跨线程消息传递
  • 2. 参数不匹配风险
    当msg.hwnd为NULL时(如线程消息),直接传参会导致窗口过程收到无效hwnd
  • 3. 破坏消息处理机制
    某些消息(如WM_PAINT)需要特殊处理时序,直接调用可能导致:
    WM_PAINT 处理不及时
    WM_TIMER 消息无法正确触发
  • 4. 模态对话框问题
    直接调用会绕过消息队列,导致模态对话框的消息循环异常


正确做法:
必须使用标准消息循环:

  1. while (GetMessage(&msg, NULL, 0, 0))
  2. {
  3.     TranslateMessage(&msg);
  4.     DispatchMessage(&msg); // 必须用这个!
  5. }
复制代码




测试现象举例:如果用直接调用WndProc:
- 按键盘时WM_CHAR可能无法触发
- 窗口重绘会出现残影
- 窗口移动时无法实时拖动

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复

使用道具 举报

发表于 2025-5-4 12:06:21 | 显示全部楼层    本楼为最佳答案   
问题:为什么不能直接调用WndProc代替DispatchMessage?


关键问题在于窗口过程的调用规范和时间戳处理!

虽然直接调用WndProc看似可行,但会引发以下问题:

  • 1. 缺少消息预处理
    DispatchMessage会在调用窗口过程前:
    - 设置线程的"上次消息时间戳"(GetMessageTime依赖这个)
    - 处理ANSI/Unicode转换
    - 处理跨线程消息传递
  • 2. 参数不匹配风险
    当msg.hwnd为NULL时(如线程消息),直接传参会导致窗口过程收到无效hwnd
  • 3. 破坏消息处理机制
    某些消息(如WM_PAINT)需要特殊处理时序,直接调用可能导致:
    WM_PAINT 处理不及时
    WM_TIMER 消息无法正确触发
  • 4. 模态对话框问题
    直接调用会绕过消息队列,导致模态对话框的消息循环异常


正确做法:
必须使用标准消息循环:

  1. while (GetMessage(&msg, NULL, 0, 0))
  2. {
  3.     TranslateMessage(&msg);
  4.     DispatchMessage(&msg); // 必须用这个!
  5. }
复制代码




测试现象举例:如果用直接调用WndProc:
- 按键盘时WM_CHAR可能无法触发
- 窗口重绘会出现残影
- 窗口移动时无法实时拖动

以上回复来自 FishC AI 小助理,如未能正确解答您的问题,请继续追问。
小甲鱼最新课程 -> https://ilovefishc.com
回复 支持 反对

使用道具 举报

您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

小黑屋|手机版|Archiver|鱼C工作室 ( 粤ICP备18085999号-1 | 粤公网安备 44051102000585号)

GMT+8, 2025-5-12 15:12

Powered by Discuz! X3.4

© 2001-2023 Discuz! Team.

快速回复 返回顶部 返回列表