百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

老生常谈C# 开发 windows 消息循环机制的原理和流程

liebian365 2024-10-26 12:59 33 浏览 0 评论

#头条创作挑战赛#

在C#开发中,我们经常会涉及到与Windows操作系统进行交互的需求。而在Windows操作系统中,消息循环机制是实现交互的基础。本文将详细介绍C#开发中的Windows消息循环机制,包括其原理和流程。

在开始之前,我们先了解一下消息循环的概念。消息循环是指在Windows操作系统中,应用程序通过不断地接收和处理消息来实现与用户的交互。当用户进行操作时,例如点击鼠标、按下键盘等,Windows会将相应的消息发送给应用程序,应用程序则通过消息循环机制来接收和处理这些消息。

Windows消息循环机制是指Windows操作系统用于接收、分发和处理各种消息的机制。它是保证Windows应用程序能够响应用户输入和系统事件的核心机制。

Windows消息循环机制的基本原理如下:

1. 创建窗口:应用程序创建一个窗口,并注册窗口过程函数(Window Procedure)来处理窗口的消息。

2. 消息循环:应用程序进入一个无限循环,不断地接收和分发消息。

3. 接收消息:操作系统将各种消息发送给目标窗口。消息可以是来自用户的输入(如鼠标点击、键盘按键),或者来自系统的通知(如定时器、窗口状态变化)等。

4. 分发消息:窗口过程函数根据消息的类型,将消息传递给相应的窗口控件或处理函数进行处理。每个窗口都有一个唯一的窗口过程函数来处理消息。

5. 处理消息:窗口控件或处理函数根据消息的具体内容,执行适当的操作。例如,对于鼠标点击消息,窗口可能会更新显示内容或触发相关的事件处理函数。

6. 返回消息:处理完消息后,窗口过程函数通常返回一个结果给操作系统,以便进一步处理。

重要的是要理解,消息循环是在应用程序的主线程中执行的。它负责接收和分发消息,然后调用窗口过程函数或控件的事件处理函数来处理这些消息。因此,应用程序需要及时地从消息循环中返回,以保持响应性,而不会阻塞主线程。

在Windows中,可以使用不同的编程框架(如Win32 API、.NET Framework、Windows Forms、WPF等)来处理消息循环。这些框架提供了相应的函数和类来简化与消息循环相关的操作,能够更加方便地处理窗口消息。

在C#开发中,我们可以使用Windows Forms或WPF等框架来创建Windows应用程序。这些框架已经为我们封装了消息循环机制,我们只需要在应用程序的主线程中调用相应的方法来启动消息循环。

下面是C#开发中Windows消息循环的详细流程:

1. 创建应用程序主窗口:首先,我们需要创建一个应用程序的主窗口,可以使用Windows Forms或WPF等框架提供的窗口类来实现。

2. 启动消息循环:在主线程中,我们需要调用Application.Run方法来启动消息循环。这个方法会一直运行,直到应用程序退出。

3. 接收消息:在消息循环中,应用程序会不断地接收消息。可以通过重写窗口类的WndProc方法来处理消息。WndProc方法是窗口类的回调函数,当有消息到达时,系统会自动调用该方法,并将消息传递给它。

4. 处理消息:在WndProc方法中,我们可以根据消息的类型进行相应的处理。例如,如果是鼠标点击消息,我们可以调用相应的方法来处理点击事件;如果是键盘按下消息,我们可以调用相应的方法来处理按键事件。

5. 分发消息:在处理完消息后,我们需要调用base.WndProc方法来分发消息。这样,其他的消息处理程序才能继续处理该消息。

6. 退出消息循环:当应用程序准备退出时,我们可以调用Application.Exit方法来退出消息循环。

需要注意的是,消息循环是一个事件驱动的过程。应用程序并不会主动去查询是否有消息到达,而是等待系统将消息送达。因此,在消息循环中,应尽量避免长时间的阻塞操作,以免影响消息的处理。

using System;
using System.Runtime.InteropServices;
using System.Windows.Forms;

class Program
{
    // 导入Windows API函数
    [DllImport("user32.dll")]
    private static extern bool GetMessage(out MSG lpMsg, IntPtr hWnd, uint wMsgFilterMin, uint wMsgFilterMax);

    [DllImport("user32.dll")]
    private static extern bool TranslateMessage([In] ref MSG lpMsg);

    [DllImport("user32.dll")]
    private static extern IntPtr DispatchMessage([In] ref MSG lpMsg);

    [DllImport("user32.dll")]
    private static extern IntPtr CreateWindowEx(
        uint dwExStyle,
        string lpClassName,
        string lpWindowName,
        uint dwStyle,
        int x,
        int y,
        int nWidth,
        int nHeight,
        IntPtr hWndParent,
        IntPtr hMenu,
        IntPtr hInstance,
        IntPtr lpParam);

    [DllImport("user32.dll")]
    private static extern bool DestroyWindow(IntPtr hWnd);

    // 定义消息结构体
    [StructLayout(LayoutKind.Sequential)]
    public struct MSG
    {
        public IntPtr hwnd;
        public uint message;
        public IntPtr wParam;
        public IntPtr lParam;
        public uint time;
        public POINT pt;
    }

    // 定义坐标结构体
    [StructLayout(LayoutKind.Sequential)]
    public struct POINT
    {
        public int X;
        public int Y;
    }

    // 定义窗口过程回调函数
    private delegate IntPtr WndProcDelegate(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam);

    private static WndProcDelegate wndProc;

    // 窗口过程回调函数
    private static IntPtr WindowProc(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam)
    {
        switch (msg)
        {
            case WM_PAINT:
                // 处理窗口重绘消息
                Console.WriteLine("窗口重绘");
                break;

            case WM_KEYDOWN:
                // 处理键盘按下消息
                Console.WriteLine("键盘按下");
                break;

            case WM_CLOSE:
                // 处理窗口关闭消息
                DestroyWindow(hWnd);
                break;

            default:
                // 其他消息交给默认处理
                return DefWindowProc(hWnd, msg, wParam, lParam);
        }

        return IntPtr.Zero;
    }

    // 创建消息循环
    private static void CreateMessageLoop()
    {
        // 注册窗口类
        WNDCLASSEX wndClass = new WNDCLASSEX();
        wndClass.cbSize = (uint)Marshal.SizeOf(wndClass);
        wndClass.lpfnWndProc = Marshal.GetFunctionPointerForDelegate(wndProc);
        wndClass.hInstance = Marshal.GetHINSTANCE(typeof(Program).Module);
        wndClass.lpszClassName = "MyWindowClass";

        if (RegisterClassEx(ref wndClass) == 0)
        {
            throw new Exception("注册窗口类失败");
        }

        // 创建窗口
        IntPtr hWnd = CreateWindowEx(
            0,
            "MyWindowClass",
            "My Window",
            WS_OVERLAPPEDWINDOW,
            CW_USEDEFAULT, CW_USEDEFAULT, 640, 480,
            IntPtr.Zero, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero);

        if (hWnd == IntPtr.Zero)
        {
            throw new Exception("创建窗口失败");
        }

        // 显示窗口
        ShowWindow(hWnd, SW_SHOWDEFAULT);

        // 进入消息循环
        MSG msg;
        while (GetMessage(out msg, IntPtr.Zero, 0, 0))
        {
            TranslateMessage(ref msg);
            DispatchMessage(ref msg);
        }

        // 销毁窗口类
        UnregisterClass("MyWindowClass", Marshal.GetHINSTANCE(typeof(Program).Module));
    }

    static void Main()
    {
        wndProc = WindowProc;
        CreateMessageLoop();
    }

    // 常量定义
    private const uint WM_PAINT = 0x000F;
    private const uint WM_KEYDOWN = 0x0100;
    private const uint WM_CLOSE = 0x0010;
    private const uint WS_OVERLAPPEDWINDOW = 0xCF0000;
    private const int CW_USEDEFAULT = unchecked((int)0x80000000);
    private const int SW_SHOWDEFAULT = 10;

    // 导入Windows API函数
    [DllImport("user32.dll")]
    private static extern short RegisterClassEx([In] ref WNDCLASSEX lpWndClass);

    [DllImport("user32.dll")]
    private static extern short UnregisterClass(string lpClassName, IntPtr hInstance);

    [DllImport("user32.dll")]
    private static extern bool ShowWindow(IntPtr hWnd, int nCmdShow);

    [DllImport("user32.dll")]
    private static extern IntPtr DefWindowProc(IntPtr hWnd, uint msg, IntPtr wParam, IntPtr lParam);

    // 定义窗口类结构体
    [StructLayout(LayoutKind.Sequential)]
    public struct WNDCLASSEX
    {
        public uint cbSize;
        public uint style;
        [MarshalAs(UnmanagedType.FunctionPtr)] public WndProcDelegate lpfnWndProc;
        public int cbClsExtra;
        public int cbWndExtra;
        public IntPtr hInstance;
        public IntPtr hIcon;
        public IntPtr hCursor;
        public IntPtr hbrBackground;
        public string lpszMenuName;
        public string lpszClassName;
        public IntPtr hIconSm;
    }
}

这个示例代码创建了一个最基本的窗口,并处理了窗口重绘、键盘按下和窗口关闭等消息。可以根据自己的需要扩展窗口过程函数中的消息处理逻辑。

请注意,在运行此示例代码之前,需要将项目设置为使用 Windows 应用程序类型,而不是控制台应用程序类型。此外,代码中调用的 user32.dll 和相关函数需要引入正确的命名空间,以确保能够正确地导入并与库进行交互。

总结起来,C#开发中的Windows消息循环机制是实现与用户交互的基础。通过创建应用程序主窗口,启动消息循环,接收和处理消息,我们可以实现丰富的交互功能。熟悉消息循环的原理和流程,对于开发Windows应用程序是非常重要的。

希望通过本文的介绍,能够更加深入地了解C#开发中的Windows消息循环机制,并能够在实际项目中灵活运用。

相关推荐

go语言也可以做gui,go-fltk让你做出c++级别的桌面应用

大家都知道go语言生态并没有什么好的gui开发框架,“能用”的一个手就能数的清,好用的就更是少之又少。今天为大家推荐一个go的gui库go-fltk。它是通过cgo调用了c++的fltk库,性能非常高...

旧电脑的首选系统:TinyCore!体积小+精简+速度极快,你敢安装吗

这几天老毛桃整理了几个微型Linux发行版,准备分享给大家。要知道可供我们日常使用的Linux发行版有很多,但其中的一些发行版经常会被大家忽视。其实这些微型Linux发行版是一种非常强大的创新:在一台...

codeblocks和VS2019下的fltk使用中文

在fltk中用中文有点问题。英文是这样。中文就成这个样子了。我查了查资料,说用UTF-8编码就行了。edit->Fileencoding->UTF-8然后保存文件。看下下边的编码指示确...

FLTK(Fast Light Toolkit)一个轻量级的跨平台Python GUI库

FLTK(FastLightToolkit)是一个轻量级的跨平台GUI库,特别适用于开发需要快速、高效且简单界面的应用程序。本文将介绍Python中的FLTK库,包括其特性、应用场景以及如何通过代...

中科院开源 RISC-V 处理器“香山”流片,已成功运行 Linux

IT之家1月29日消息,去年6月份,中科院大学教授、中科院计算所研究员包云岗,发布了开源高性能RISC-V处理器核心——香山。近日,包云岗在社交平台晒出图片,香山芯片已流片,回片后...

Linux 5.13内核有望合并对苹果M1处理器支持的初步代码

预计Linux5.13将初步支持苹果SiliconM1处理器,不过完整的支持工作可能还需要几年时间才能完全完成。虽然Linux已经可以在苹果SiliconM1上运行,但这需要通过一系列的补丁才能...

Ubuntu系统下COM口测试教程(ubuntu port)

1、在待测试的板上下载minicom,下载minicom有两种方法:方法一:在Ubuntu软件中心里面搜索下载方法二:按“Ctrl+Alt+T”打开终端,打开终端后输入“sudosu”回车;在下...

湖北嵌入式软件工程师培训怎么选,让自己脱颖而出

很多年轻人毕业即失业、面试总是不如意、薪酬不满意、在家躺平。“就业难”该如何应对,参加培训是否能改变自己的职业走向,在湖北,有哪些嵌入式软件工程师培训怎么选值得推荐?粤嵌科技在嵌入式培训领域有十几年经...

新阁上位机开发---10年工程师的Modbus总结

前言我算了一下,今年是我跟Modbus相识的第10年,从最开始的简单应用到协议了解,从协议开发到协议讲解,这个陪伴了10年的协议,它一直没变,变的只是我对它的理解和认识。我一直认为Modbus协议的存...

创建你的第一个可运行的嵌入式Linux系统-5

@ZHangZMo在MicrochipBuildroot中配置QT5选择Graphic配置文件增加QT5的配置修改根文件系统支持QT5修改output/target/etc/profile配置文件...

如何在Linux下给zigbee CC2530实现上位机

0、前言网友提问如下:粉丝提问项目框架汇总下这个网友的问题,其实就是实现一个网关程序,内容分为几块:下位机,通过串口与上位机相连;下位机要能够接收上位机下发的命令,并解析这些命令;下位机能够根据这些命...

Python实现串口助手 - 03串口功能实现

 串口调试助手是最核心的当然是串口数据收发与显示的功能,pzh-py-com借助的是pySerial库实现串口收发功能,今天痞子衡为大家介绍pySerial是如何在pzh-py-com发挥功能的。一、...

为什么选择UART(串口)作为调试接口,而不是I2C、SPI等其他接口

UART(通用异步收发传输器)通常被选作调试接口有以下几个原因:简单性:协议简单:UART的协议非常简单,只需设置波特率、数据位、停止位和校验位就可以进行通信。相比之下,I2C和SPI需要处理更多的通...

同一个类,不同代码,Qt 串口类QSerialPort 与各种外设通讯处理

串口通讯在各种外设通讯中是常见接口,因为各种嵌入式CPU中串口标配,工业控制中如果不够还通过各种串口芯片进行扩展。比如spi接口的W25Q128FV.对于软件而言,因为驱动接口固定,软件也相对好写,因...

嵌入式linux为什么可以通过PC上的串口去执行命令?

1、uboot(负责初始化基本硬bai件,如串口,网卡,usb口等,然du后引导系统zhi运行)2、linux系统(真正的操作系统)3、你的应用程序(基于操作系统的软件应用)当你开发板上电时,u...

取消回复欢迎 发表评论: