走进windows编程的世界-----消息处理函数(1)

Win32消息机制

过程驱动:程序是按照我们预先定义好的顺序执行,每执行一步,下一步都已经按照预定的顺序 继续执行,直至程序结束。

事件驱动:程序的执行顺序是无序的。某个时间点所执行的代码,是由外界 通知。由于我们无法决定程序执行顺序。所以代码的执行也是无序的。

Win32基本消息

WM_DESTROY:
      窗口销毁时的消息,可以做退出或善后处理

WM_CREATE:
      窗口创建消息,是在窗口创建后,窗口处理函数收到的第一条消息
  可以在这个消息内,做初始化或者穿件子窗口
  WPARAM wParam - 不使用
  LPARAM lParam - CREATESTRUCT指针
   WM_SIZE:
      当窗口大小发生改变时,会收到这个消息。
  可以在这个消息中调整窗口的布局
 WM_SYSCOMMAND:
      系统命令消息,当点击系统菜单和按钮时会收到
  可以在这个消息中,提示用户保存数据等
 WM_PAINT:
      绘图消息
      键盘消息:
 鼠标消息
 WM_TIME:定时器消息

消息的获取和发送

获取GetMessage/PeekMessage
         GetMessage 获取消息,阻塞函数
         PeekMessage 获取消息,非阻塞函数
     发送SendMessage/PostMessage
         SendMessage 发送消息并等候消息处理结束才返回。 
         PostMessage 发送消息后立即返回,不关心消息处理的结果。

LRESULT SendMessage/PostMessage(
          HWND hWnd,      //处理消息窗口
          UINT Msg,       //消息的ID
          WPARAM wParam,  //消息的参数
          LPARAM lParam );//消息的参数

3 消息组成和分类
   3.1 消息组成
      窗口句柄/消息ID/消息参数(WPARAM.LPARAM)
   3.2 消息分类
      3.2.1 系统消息 - 由系统定义和使用的消息
         例如:WM_CREATE/WM_SIZE
         消息ID范围为: 0 - 0x03FF(WM_USER-1)
      3.2.2 用户定义消息 - 应用程序可以自己定义和使用的消息, WM_USER(0x0400)
         从WM_USER的ID开始,到0x7FFF,是用户可以定义使用的消息.
      3.2.3 其他消息范围
         WM_APP(0x8000)-0xBFFF:应用程序访问窗口的消息ID
         0xC000-0xFFFF: 应用程序访问消息,使用字符串注册系统产生相应消息ID
      3.2.4 用户定义消息的使用
         1)定义自定义消息ID:

 #define   WM_FIRSTMSG  (WM_USER+1)

2)在窗口处理函数中,响应消息

 switch( nMsg )
 {
 case WM_FIRSTMSG:
   //处理函数
    break;
 }

3)SendMessage/PostMessage发送消息

    SendMessage( hWnd, WM_FIRSTMSG, 0, 0 );

4 消息队列
    4.1 消息队列 - 用于存储消息的内存空间,消息在队列中是先入先出.
    4.2 消息队列的分类
      4.2.1 系统消息队列 - 由系统维护的消息队列. 
      4.2.2 应用程序消息队列(线程消息对列) -属于每个线程的各自拥有的消息队列.
  
  5 消息和消息队列
    5.1 根据消息和消息队列关系,将消息分成两种:
      队列消息 - 可以存放在消息队列中的消息.
      非队列消息 - 发送时不进入消息队列.
    5.2 队列消息
      首先存放到消息队列当中,然后由GetMessage/PeekMessage取出,然后进行处理.
      例如: 鼠标消息/键盘消息/WM_PAINT/WM_QUIT/M_TIMER消息
    5.3 非队列消息
      消息发送直接发送给指定的窗口,查找窗口的处理函数,返回处理结果.
 
 6 消息的获取   
    6.1 消息循环
      6.1.1 GetMesssage从队列中获取消息,判断是否是WM_QUIT消息,如果发现是WM_QUIT消息,消息循环结束,否则继续下一步.
      6.1.2 TranslateMessage 翻译按键消息,如果发现有按键消息,产生字符消息放入消息对列, 继续下一步
      6.1.3 DispatchMessage 找到消息所发窗口的处理函数,处理消息.处理完成后,返回6.1.1.
    6.2 GetMesssage和PeekMessage
      6.2.1 从线程消息队列中获取消息,如果找到消息,就返回消息,进行消息处理. 如果未找到消息,执行6.2.2
      6.2.2 查找系统消息队列.通过向系统消息队列查询,如果找到消息,获取消息并返回,进行消息处理.如果未找到消息,执行6.2.3
      6.2.3 检查窗口需要重新绘制的范围,如果发现存在重新绘制的范围,会产生WM_PAINT消息.然后进行消息处理, 如果未找,执行6.2.4
      6.2.4 检查WM_TIMER定时器消息,如果发现存在已经到时的定时器,会产生WM_TIMER消息.进行消息处理. 如果未找,执行6.2.5
      6.2.5 执行内存管理工作.
      6.2.6 根据函数不同,处理结果不同:
        GetMesssage - 阻塞,等候下一条消息
        PeekMessage - 让出控制权,交给后面的代码执行.
        
  7 消息发送  
    7.1 消息发送分两种
       发送(Send)消息 - 直接发送给指定的窗口,并等候结果.
       投递(Post)消息 - 发送到消息队列当中,立刻返回,由消息循环处理.
    7.2 PostMessage和SendMessage
      PostMessage产生队列消息,由于发送后不等候消息处理结果,所以不能确定消息是否被处理成功.
      SendMessage产生非队列消息,可以确定消息是否成功.

看下面的代码示例:

/*File : message.cpp
 *Auth : sjin
 *Date : 20140519
 *Mail : [email protected]
 */

#include <iostream>
#include <Windows.h>
#include <stdio.h>

HINSTANCE g_hInst = NULL;
HWND g_button = NULL;

LRESULT CALLBACK WndProc(HWND hWnd,
                         UINT nMsg,
                         WPARAM wParam,
                         LPARAM lParam);

/*注册窗口*/
BOOL RegisterWnd(LPSTR pszClassName)
{
    WNDCLASSEX wce = {0};
    wce.cbClsExtra = 0;
    wce.cbSize = sizeof(wce);
    wce.cbWndExtra = 0;
    wce.hbrBackground = HBRUSH(COLOR_BTNFACE+1);
    wce.hCursor = NULL;
    wce.hIcon = NULL;
    wce.hIconSm = NULL;
    wce.hInstance = g_hInst;
    wce.lpfnWndProc = WndProc;
    wce.lpszClassName = pszClassName;
    wce.lpszMenuName = NULL;
    wce.style = CS_HREDRAW|CS_VREDRAW;

    ATOM nAtom = RegisterClassEx(&wce);
    if(0 == nAtom)
        return FALSE;
    return TRUE;
}

/*创建窗口*/
HWND CreateWnd(LPSTR pszClassName)
{
    HWND hWnd = CreateWindowEx(0, pszClassName,
        "MyWnd", WS_OVERLAPPEDWINDOW, CW_USEDEFAULT,
        CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,
        NULL,NULL,g_hInst,NULL);
    return hWnd;
}

 /*显示窗口*/
void DisplayWnd(HWND hWnd)
{
    ShowWindow(hWnd, SW_SHOW);
    UpdateWindow(hWnd);
}

void wm_create(HWND hWnd,UINT nMsg,
	           WPARAM wParam,LPARAM lParam)
{
	LPCREATESTRUCT pCreateStruct = LPCREATESTRUCT(lParam);
	/*打印窗口的名字
	  先弹出这个对话框,点击OK后弹出窗口
	*/
	//MessageBox(NULL, pCreateStruct->lpszName,"WM_CREATE",MB_OK);

	/*创建一个子窗口*/
	 g_button = CreateWindowEx(0, "BUTTON",
		"BUTTON", WS_CHILD|WS_VISIBLE, 30,
        20,100,50,
        hWnd,NULL,g_hInst,NULL);

}

void wm_size(HWND hWnd,UINT nMsg,
	           WPARAM wParam,LPARAM lParam)
{
	INT nWidth = LOWORD(lParam);
	INT nHeight = HIWORD(lParam);
	CHAR szText[256] = {‘\0‘};
	sprintf(szText,"W:%d;H:%d",nWidth,nHeight);
	//MessageBox(NULL, szText,"WM_SIZE",MB_OK);

	 if(NULL != g_button)
    {
        int nX = (nWidth - 100)/2;
        int nY = (nHeight - 100)/2;
        MoveWindow(g_button, nX, nY, 100, 100, TRUE);
    }
}

/*执行系统命令函数处理*/
BOOL wm_syscommand(HWND hWnd,UINT nMsg,
	           WPARAM wParam,LPARAM lParam)
{
	switch(wParam){
	case SC_CLOSE:
		if(IDOK == MessageBox(NULL,"是否将文件存盘","提示",MB_OKCANCEL|MB_ICONWARNING)){
			return TRUE;
		} else {
			return FALSE;
		}
		break;
	default:
		break;
	}

	return FALSE;
}

/*消息处理函数*/
LRESULT CALLBACK WndProc(HWND hWnd,
                         UINT nMsg,
                         WPARAM wParam,
                         LPARAM lParam)
{
    switch(nMsg){
	/*Win 32 基本消息
	  WM_DESTROY:
	       窗口销毁时的消息,可以做退出或善后处理
	  WM_CREATE:
	       窗口创建消息,是在窗口创建后,窗口处理函数收到的第一条消息
		   可以在这个消息内,做初始化或者穿件子窗口
		   WPARAM wParam - 不使用
		   LPARAM lParam - CREATESTRUCT指针
	  WM_SIZE:
	       当窗口大小发生改变时,会收到这个消息。
		   可以在这个消息中调整窗口的布局
	  WM_SYSCOMMAND:
	       系统命令消息,当点击系统菜单和按钮时会收到
		   可以在这个消息中,提示用户保存数据等
	  WM_PAINT:
	       绘图消息
      键盘消息:
	  鼠标消息
	  WM_TIME:定时器消息
	*/
    case WM_DESTROY://窗口销毁时的消息
		/*向窗口发送WM_QUIT 消息*/
        //SendMessage(hWnd, WM_QUIT, 0, 0);/*发送消息并等候消息处理结束才返回*/
        PostMessage(hWnd, WM_QUIT, 0, 0);/*发送消息后立即返回,不关心消息处理的结果*/
        //PostQuitMessage(0);
		return 0;
	case WM_CREATE:/*创建窗口*/
		wm_create(hWnd,nMsg,wParam,lParam);
		break;

	case WM_SIZE:/*窗口拖动*/
		wm_size(hWnd,nMsg,wParam,lParam);
		break;
	case WM_SYSCOMMAND:/*执行系统命令*/
		if(!wm_syscommand(hWnd,nMsg,wParam,lParam)){
			return 0;
		}
		break;

    }
    return DefWindowProc(hWnd, nMsg, wParam, lParam);
}

/*消息循环*/
void Message()
{
	/* MSG 结构体参数描述
	typedef struct tagMSG {     // msg
       HWND hwnd; //消息窗口句柄
       UINT message;//消息标示
       WPARAM wParam;//消息的参数
       LPARAM lParam;//消息的参数
       DWORD time;//消息的时间
       POINT pt;//消息产生时,鼠标的位置
     } MSG;
    */

    MSG msg = {0};

	/*
	  BOOL GetMessage(
      LPMSG lpMsg,//存放获取到的消息数据 由系统填写关于消息的参数
      HWND hWnd,//获取消息的窗口句柄,可接受指定窗口消息
      UINT wMsgFilterMin,//消息过滤的起始消息
      UINT wMsgFilterMax //消息过滤的终止消息
	  );
	   return :
	        TRUE :成功获取消息;FALSE:获取到WM_QUIT消息时。
	   可以使用PostQuitMessage向窗口发送WM_QUIT消息
	   GetMessage:获取消息,阻塞函数
	   PeekMessage:获取消息,非阻塞函数
	 */
    while(GetMessage(&msg, NULL, 0, 0))
    {
		/*TranslateMessage:
		  就是将键盘的消息转换成字符消息
		  1、首先检查是否是键盘消息
		  2、如果发现是按键消息,将根据按键,产生字符消息
		     在下一个GetMessage执行时,收到这个消息
		  3、如果未发现按键消息,未做任何处理
		*/
        TranslateMessage(&msg);
		/*DispatchMessage
		  根据消息数据内窗口句柄,找到这个窗口的窗口处理函数, 调用处理函数,进行消息处理。
          如果MSG中,HWND窗口句柄为空,DispatchMessage不做任何处理。
		*/
        DispatchMessage(&msg);
    }
}

int APIENTRY WinMain(HINSTANCE hInstance,
                     HINSTANCE hPrevInstance,
                     LPSTR     lpCmdLine,
                     int       nCmdShow)
{
    g_hInst = hInstance;
    RegisterWnd("MyWnd");
    HWND hWnd = CreateWnd("MyWnd");
    DisplayWnd(hWnd);
    Message();
    return 0;
}

走进windows编程的世界-----消息处理函数(1),布布扣,bubuko.com

时间: 2024-08-05 19:37:46

走进windows编程的世界-----消息处理函数(1)的相关文章

走进windows编程的世界-----消息处理函数(4)

一 右键菜单  1 右键菜单    当在窗口点击鼠标右键时,弹出的菜单.  2 右键菜单的使用    2.1 创建菜单      CreatePopupMenu    2.2 菜单增加     AppendMenu    2.3 菜单的显示. BOOL TrackPopupMenu( HMENU hMenu, //显示的菜单句柄 UINT uFlags, //显示的方式 int x, //菜单的X屏幕坐标 int y, //菜单的Y屏幕坐标 int nReserved, //保留,必须为0 HW

走进windows编程的世界-----消息处理函数(3)

二 定时器消息 1 定时器消息 WM_TIMER   按照定时器设置时间段,自动向窗口发送一个定时器消息WM_TIMER. 优先级比较低.   定时器精度比较低,毫秒级别.消息产生时间也精度比较低.    2 消息和函数   2.1 WM_TIMER  - 消息ID    wParam: 定时器的ID    lParam: 定时器的处理函数 2.2 SetTimer  - 设置一个定时器 UINT SetTimer( HWND hWnd, //窗口的句柄,可以为NULL UINT nIDEven

走进windows编程的世界-----消息处理函数(2)

一 WM_PAINT消息 1 WM_PAINT的产生   由于窗口的互相覆盖等,产生需要绘制的区域,那么会产生WM_PAINT消息.   一般情况下,不直接发送WM_PAINT消息,通过API声明需要绘制区域,来产生WM_PAINT消息.   例如,可以使用InvalidateRect声明一个需要重新绘制的区域.    2 WM_PAINT的注意点    2.1 如果一个消息队列中,有多个WM_PAINT消息,只有最后一个WM_PAINT消息会被处理.    2.2 WM_PAINT消息处理中,

走进windows编程的世界-----窗口的注册及创建

1   窗口注册和创建 1.1WIN32 窗口程序创建步骤 1.WinMain入口函数的定义 2.WindowProc函数的定义 3.注册窗口类 RegisterClass.RegisterClassEX 4.创建窗口 CreateWindow.CreateWindowEx HWND CreateWindow( LPCTSTRlpClassName,//指向已注册的窗口类的名称的指针 LPCTSTRlpWindowName,//指向窗口名称的指针 DWORDdwStyle,//窗口的风格 int

走进windows编程的世界-----windows进程

Windows进程  1 Windows进程    进程是一个容器,包含了一个应用程序实例的各种资源.Windows多任务的操作系统,因此可以同时执行多个进程.      2 Windows进程的一些特点    2.1 进程中包含了执行代码等资源.    2.2 进程都具有私有的地址空间.    2.3 每个进程都有一个ID,标识进程.    2.4 每个进程都有自己的安全属性    2.5 至少要包含一个可以执行的线程.    二 进程的环境 1 环境信息的获取    获取:    LPVOI

走进windows编程的世界-----字符编码

1   字符编码 1.1编码的历史 1.1.1ASCII码 0=127 7位表示 1.1.2ASCII扩展码 0-255 8为表示. 代码页:通过代码也来切换对应的字符(数字表示) 1.1.3双字节字符集DBCS 使用一个或两个字节表示字符. 1.1.4Unicode编码 全部使用2个字节表示字符 内存 硬盘等资源占用变大.对编码支持度大. 字符集 1.2C 语言和编码 1.2.1单字节的字符和字符串 Char  cText = 'A'; Char * pszText ="ABCD"

走进windows编程的世界-----绘图相关

Windows绘图 1 图形绘制      1.1 图形绘制的方式      获取到绘图句柄-设备描述表(DC),使用相应的绘图的API,在设备上绘制图形.          1.2 颜色      R\G\B三色, 每种颜色8位, 共24位颜色.      32位颜色: 颜色数量24位颜色, 多出来的8位表示灰度.      16位: 颜色数量2的16次方.            Win32下,颜色的定义 COLORREF(DWORD), RGB宏定义颜色       COLORREF nCo

走进windows编程的世界-----位图及映射模式

1   位图的使用 1.1位图介绍 位图-通过保存在图像上每个点的颜色,生成响应的位图文件. 光栅图:图像点阵的保存 矢量图:绘图命令的保存. 1.2位图的使用 1.             加载位图资源 LoadBitap(); 2.             创建防止位图的DC CreateCompatibleDC 3.             将位图放入创建的DC SlectObject 4.             绘制位图到当前DC中 BitBlt 5.             取出位图

走进windows编程的世界-----入门篇

1   Windows编程基础 1.1Win32应用程序基本类型 1)  控制台程序 不须要完好的windows窗体,能够使用DOS窗体方式显示 2)  Win32窗体程序 包括窗体的程序,能够通过窗体与程序进行交互 3)  Win32库程序 提供已有的代码,供其它程序使用 动态库(DLL):是在运行的时候能够载入的. 静态库(LIB):是在编译链接是使用的程序.成为当前程序的一部分. 1.2头文件和库 1.2.1头文件 主要的头文件windows.h包括了windows经常使用的定义等,其它,