Windows 操作系统定义了大量的消息类型

Windows 操作系统定义了大量的消息类型,每种消息类型都有其特定的含义和用途。Windows 消息是操作系统与应用程序之间进行通信的主要机制,它们用于通知应用程序各种事件的发生,例如用户输入(键盘、鼠标)、窗口状态变化、系统事件等。

常见的 Windows 消息类型

下面列出了几种常见的 Windows 消息类型及其用途:

1. 窗口管理消息

这些消息用于管理窗口的生命周期、大小、位置和状态等。

  • WM_CREATE:窗口正在被创建时发送,通常用于进行初始化。
  • WM_DESTROY:窗口即将被销毁时发送,通常用于清理资源。
  • WM_CLOSE:请求关闭窗口时发送,例如用户点击窗口的关闭按钮。
  • WM_QUIT:指示应用程序应终止。通常由 PostQuitMessage 发送。
  • WM_SIZE:窗口大小改变时发送,应用程序可以在此消息中调整窗口的布局。
  • WM_MOVE:窗口被移动时发送。
2. 鼠标输入消息

这些消息用于处理鼠标事件,例如移动、点击、滚动等。

  • WM_MOUSEMOVE:鼠标移动时发送。
  • WM_LBUTTONDOWN:鼠标左键按下时发送。
  • WM_LBUTTONUP:鼠标左键释放时发送。
  • WM_RBUTTONDOWN:鼠标右键按下时发送。
  • WM_RBUTTONUP:鼠标右键释放时发送。
  • WM_MOUSEWHEEL:鼠标滚轮滚动时发送。
3. 键盘输入消息

这些消息用于处理键盘事件,例如按键按下和释放。

  • WM_KEYDOWN:按下一个键时发送。
  • WM_KEYUP:释放一个键时发送。
  • WM_CHAR:按键被按下并转换为字符时发送。
4. 绘制和重绘消息

这些消息用于处理窗口的绘制和重绘操作。

  • WM_PAINT:需要重绘窗口时发送,例如窗口首次显示时或被覆盖的窗口区域重新出现时。
  • WM_ERASEBKGND:窗口背景需要被擦除时发送,通常在处理 WM_PAINT 消息前发送。
5. 定时器消息

这些消息用于处理定时事件。

  • WM_TIMER:定时器触发时发送,应用程序可以使用 SetTimer 设置定时器并响应此消息。
6. 剪贴板消息

这些消息用于处理剪贴板操作。

  • WM_CUT:执行剪切操作时发送。
  • WM_COPY:执行复制操作时发送。
  • WM_PASTE:执行粘贴操作时发送。
7. 系统命令消息

这些消息用于处理系统菜单命令和其他系统相关操作。

  • WM_SYSCOMMAND:用户选择窗口的系统菜单命令(例如,最小化、最大化、关闭)时发送。
8. 控制和对话框消息

这些消息用于处理控件(如按钮、编辑框等)和对话框的交互。

  • WM_COMMAND:用于通知控件事件(如按钮点击),常用于菜单命令和控件事件。
  • WM_NOTIFY:用于处理控件的通知消息,例如列表视图的状态变化。
  • WM_HSCROLLWM_VSCROLL:处理水平和垂直滚动条的消息。
9. 系统级消息

这些消息通常涉及整个系统或非特定窗口的事件。

  • WM_POWERBROADCAST:系统电源状态改变时发送,例如系统进入睡眠模式。
  • WM_DEVICECHANGE:系统中设备状态改变时发送,例如插入或移除 USB 设备。

消息处理的机制

Windows 应用程序通过一个称为“窗口过程函数”的函数来处理这些消息。窗口过程函数通常是一个 switch 语句,根据接收到的消息类型进行相应的处理。例如:

LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) {
    switch (msg) {
        case WM_CREATE:
            // 初始化代码
            break;
        case WM_PAINT:
            // 绘制代码
            break;
        case WM_DESTROY:
            PostQuitMessage(0);
            break;
        default:
            return DefWindowProc(hwnd, msg, wParam, lParam);
    }
    return 0;
}

在此函数中,msg 参数表示消息类型。应用程序根据消息类型调用适当的处理代码。如果消息类型不在 switch 语句中处理,通常会调用 DefWindowProc 函数进行默认处理。

总结

Windows 消息类型众多,每种消息类型都对应特定的事件或行为。理解和处理这些消息是开发 Windows 桌面应用程序的基础。程序通过消息循环不断获取和分派消息,并在窗口过程函数中根据消息类型执行相应的操作。

----------

深入了解 Windows 消息机制和具体消息类型的作用有助于开发更加高效和响应式的 Windows 应用程序。以下将进一步探讨 Windows 消息机制的核心概念和几种常用消息的详细解释。

Windows 消息机制的核心概念

Windows 消息机制是操作系统和应用程序之间沟通的桥梁。通过这种机制,Windows 能够通知应用程序关于用户输入、系统事件以及窗口状态变化等各种事件。

消息的生命周期
  1. 消息的生成

    • 用户输入:如键盘输入、鼠标点击、鼠标移动等。
    • 系统事件:如窗口重绘请求、系统命令、设备变化、电源状态变化等。
    • 应用程序事件:如定时器到期、自定义消息等。
  2. 消息的排队

    • Windows 将生成的消息放入相应线程的消息队列中。每个 GUI 线程都有一个自己的消息队列,用于存放与该线程相关的消息。
  3. 消息的获取和分派

    • 应用程序通过消息循环(GetMessageTranslateMessageDispatchMessage)获取消息,并将其发送到合适的窗口过程函数进行处理。
  4. 消息的处理

    • 在窗口过程函数中,应用程序根据消息类型决定如何处理消息,执行相应的代码逻辑。
  5. 消息的结束

    • 部分消息处理后不会再进行额外操作,例如绘制窗口、处理用户输入等;而其他消息可能会被传递给默认窗口过程函数 DefWindowProc 进行默认处理。

深入探讨常见消息类型

1. 窗口管理消息

这些消息主要用于管理窗口的生命周期和状态。

  • WM_CREATE:

    • 含义:在窗口被创建时发送。此消息在 CreateWindowCreateWindowEx 函数返回之前发送,表示窗口的初始化开始。
    • 用途:常用于初始化窗口控件、分配资源等。例如,创建一个子窗口或控件。
    • 处理方式:在窗口过程函数中,可以使用 lParam 参数指向的 CREATESTRUCT 结构体获取创建窗口时传递的附加数据。
  • WM_DESTROY:

    • 含义:窗口即将被销毁时发送。此消息通常在用户关闭窗口或者调用 DestroyWindow 函数时触发。
    • 用途:用于释放在 WM_CREATE 或其他地方分配的资源,例如销毁控件、关闭文件、释放内存等。
    • 处理方式:调用 PostQuitMessage(0) 函数来退出消息循环,关闭应用程序。
  • WM_CLOSE:

    • 含义:请求关闭窗口时发送。例如,用户点击窗口的关闭按钮或按下 Alt+F4
    • 用途:用于执行关闭前的清理操作,如提示用户保存数据或确认退出。
    • 处理方式:可以调用 DestroyWindow(hwnd) 来销毁窗口,也可以拦截消息以取消关闭操作。
2. 鼠标输入消息

这些消息用于处理鼠标事件,帮助应用程序响应用户的鼠标操作。

  • WM_MOUSEMOVE:

    • 含义:当鼠标移动到窗口的客户区内时发送。
    • 用途:用于更新鼠标坐标、显示提示信息、改变鼠标光标外观等。
    • 处理方式lParam 包含鼠标的 x 和 y 坐标,可以通过 LOWORD(lParam)HIWORD(lParam) 获取。
  • WM_LBUTTONDOWNWM_LBUTTONUP:

    • 含义:分别表示鼠标左键按下和释放事件。
    • 用途:用于检测鼠标点击事件,执行相应的操作,如选择、拖放、按钮点击等。
    • 处理方式wParam 包含键盘状态(如 MK_CONTROLMK_SHIFT),lParam 包含鼠标坐标。
  • WM_RBUTTONDOWNWM_RBUTTONUP:

    • 含义:分别表示鼠标右键按下和释放事件。
    • 用途:通常用于显示上下文菜单或执行其他右键操作。
    • 处理方式:类似于 WM_LBUTTONDOWNWM_LBUTTONUP,处理鼠标右键事件。
  • WM_MOUSEWHEEL:

    • 含义:鼠标滚轮滚动时发送。
    • 用途:用于滚动视图、缩放内容等。
    • 处理方式wParam 高位字包含滚轮增量(每个滚动单位 120),低位字表示键盘状态;lParam 包含鼠标光标的 x 和 y 坐标。
3. 键盘输入消息

这些消息用于处理键盘事件,让应用程序能够响应用户的键盘操作。

  • WM_KEYDOWNWM_KEYUP:

    • 含义:分别表示按键按下和释放事件。
    • 用途:用于处理键盘输入,例如快捷键、游戏控制、文本输入等。
    • 处理方式wParam 包含虚拟键码,lParam 包含按键状态信息(如重复计数、扩展键标志)。
  • WM_CHAR:

    • 含义:键盘输入被转换为字符时发送。它与 WM_KEYDOWN 的区别在于 WM_CHAR 代表字符,而 WM_KEYDOWN 代表虚拟键。
    • 用途:用于文本输入,处理字符信息而不是物理按键。例如,在一个文本编辑器中处理字符输入。
    • 处理方式wParam 包含字符代码(如 Unicode 字符),lParam 包含键盘状态信息。
4. 绘制和重绘消息

这些消息用于管理窗口的绘制和重绘过程。

  • WM_PAINT:

    • 含义:窗口客户区需要重绘时发送,如窗口首次显示、被覆盖部分重新出现、窗口大小改变等。
    • 用途:用于重绘窗口内容,确保显示正确的界面。
    • 处理方式:通常调用 BeginPaint 函数开始绘制,使用 GDI 函数绘制内容后,调用 EndPaint 函数结束绘制。
  • WM_ERASEBKGND:

    • 含义:在 WM_PAINT 消息处理之前发送,用于擦除窗口背景。
    • 用途:用于清除旧的绘制内容,避免图像残留。
    • 处理方式:应用程序可以选择处理此消息,返回一个非零值表示背景已被擦除,或者调用 DefWindowProc 使用默认背景擦除。
5. 系统命令消息

这些消息用于响应系统相关的操作,如系统菜单命令、窗口最小化和最大化等。

  • WM_SYSCOMMAND:
    • 含义:当用户选择系统菜单中的命令或使用快捷键(如 Alt + 空格)时发送。
    • 用途:处理系统级命令,如最小化、最大化、关闭窗口等。
    • 处理方式wParam 的低位字包含系统命令标识符(如 SC_CLOSESC_MINIMIZESC_MAXIMIZE),lParam 通常未使用。

消息的处理顺序

Windows 应用程序通常会在一个窗口过程中处理多种类型的消息。典型的消息处理顺序如下:

  1. WM_CREATE:初始化窗口或控件。
  2. WM_PAINT:绘制或重绘窗口内容。
  3. WM_KEYDOWN / WM_CHAR / WM_KEYUP:处理键盘输入。
  4. WM_MOUSEMOVE / WM_LBUTTONDOWN / WM_LBUTTONUP:处理鼠标输入。
  5. WM_CLOSE:处理窗口关闭请求。
  6. WM_DESTROY:执行清理工作,准备退出程序。

深入理解窗口过程函数

窗口过程函数(WndProc)是处理所有消息的核心。每一个与窗口关联的消息都会通过 WndProc 函数进行处理。它是一个回调函数,具有以下典型的定义:

LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) {
    switch (msg) {
        case WM_CREATE:
            // 初始化窗口资源
            break;
        case WM_PAINT:
            // 绘制窗口
            break;
        case WM_COMMAND:
            // 处理菜单或控件命令
            break;
        case WM_DESTROY:
            PostQuitMessage(0); // 发送退出消息
            break;
       

 

#include <windows.h>
#include <stdio.h>

// 窗口过程函数
LRESULT CALLBACK WndProc(HWND hwnd, UINT msg, WPARAM wParam, LPARAM lParam) {
    switch (msg) {
        case WM_LBUTTONDOWN: { // 处理左键点击
            int xPos = LOWORD(lParam);  // 获取鼠标x坐标
            int yPos = HIWORD(lParam);  // 获取鼠标y坐标
            char buf[100];
            snprintf(buf, sizeof(buf), "Left button clicked at (%d, %d)", xPos, yPos);
            SetWindowText(hwnd, buf);  // 更新窗口标题栏为点击位置
            break;
        }

        case WM_RBUTTONDOWN: { // 处理右键点击
            int xPos = LOWORD(lParam);  // 获取鼠标x坐标
            int yPos = HIWORD(lParam);  // 获取鼠标y坐标
            char buf[100];
            snprintf(buf, sizeof(buf), "Right button clicked at (%d, %d)", xPos, yPos);
            SetWindowText(hwnd, buf);  // 更新窗口标题栏为点击位置
            break;
        }

        case WM_MOUSEMOVE: { // 处理鼠标移动
            int xPos = LOWORD(lParam);  // 获取鼠标x坐标
            int yPos = HIWORD(lParam);  // 获取鼠标y坐标
            char buf[100];
            snprintf(buf, sizeof(buf), "Mouse moved to (%d, %d)", xPos, yPos);
            SetWindowText(hwnd, buf);  // 更新窗口标题栏为鼠标移动位置
            break;
        }

        case WM_DESTROY:
            PostQuitMessage(0); // 退出消息循环
            break;

        default:
            return DefWindowProc(hwnd, msg, wParam, lParam); // 默认窗口过程
    }
    return 0;
}

// 程序入口
int APIENTRY WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow) {
    const char CLASS_NAME[] = "Sample Window Class";

    // 定义窗口类
    WNDCLASS wc = {};
    wc.lpfnWndProc = WndProc;
    wc.hInstance = hInstance;
    wc.lpszClassName = CLASS_NAME;
    wc.hCursor = LoadCursor(NULL, IDC_ARROW);  // 设置光标

    // 注册窗口类
    if (!RegisterClass(&wc)) {
        MessageBox(NULL, "Window Class Registration Failed!", "Error", MB_OK | MB_ICONERROR);
        return 0;
    }

    // 创建窗口
    HWND hwnd = CreateWindowEx(
        0,                              // 可选窗口样式
        CLASS_NAME,                     // 窗口类名
        "Sample Window",                // 窗口标题
        WS_OVERLAPPEDWINDOW,            // 窗口样式
        CW_USEDEFAULT, CW_USEDEFAULT,   // 初始位置
        CW_USEDEFAULT, CW_USEDEFAULT,   // 初始尺寸
        NULL,                           // 父窗口
        NULL,                           // 菜单
        hInstance,                      // 实例句柄
        NULL                            // 附加应用程序数据
    );

    if (hwnd == NULL) {
        MessageBox(NULL, "Window Creation Failed!", "Error", MB_OK | MB_ICONERROR);
        return 0;
    }

    // 显示窗口
    ShowWindow(hwnd, nCmdShow);
    UpdateWindow(hwnd);

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

    return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值