windows程序基础

一、windows程序基础

1. Windows程序的特点

1)用户界面统一、友好

2)支持多任务:允许用户同时运行多个应用程序(窗口)

3)独立于设备的图形操作

使用图形设备接口( GDI, Graphics Device Interface )屏蔽了不同硬件设备的差异,提供了设备无关的图形输出能力

4)队列化消息输入,支持队列特征的消息驱动模型

把从键盘、鼠标等输入设备接收的输入信息放入应用程序的队列中; Windows操作系统管理队列;应用程序需要输入信息时,不读硬件端口,只读消息队列。

5)事件驱动的程序设计

  • Windows程序由事件的发展控制,允许程序的用户用各种合理的顺序来安排程序的流程。

6)资源共享

-各个应用程序共享系统提供的资源,包括设备上下文、画刷、画笔、字体、对话框、图标、定时器、通信端口。

7)程序和资源分开

-菜单、对话框、位图等可视对象被单独分离出来加公的,全部资源定义都放在资源文件中(通常以.rc为后缀名)

8)支持应用程序间的数据交换

2. 一个简单的Windows应用程序
cpp 复制代码
#include <windows.h>
int WINAPI WinMain (HINSTANCE hInstance, HINS TANCE
hPrevInstance, LPSTR IpCmdLine, int nCmdShow)
{
MessageBox (NULL, "你好,我的Visual C++世界! ", "问候",MB_ OKCANCEL);
return 0 ;
}

运行结果:

3.事件及事件驱动

●Windows系统也叫事件驱动的系统。当用户按下一个键、移动鼠标或单击鼠标按钮时,计算机通知Windows系统已经发生了一一个事件,以及事件的种类、发生的时间、发生的位置(如坐标值)。

●事件通常以下列三种方式产生:

-通过输入设备,如键盘和鼠标;

-通过屏幕上可视的对象,如菜单、工具栏按钮、动条和对话框上的控件;

-来自Windows内部,如当一个后面的窗口显示到窗口前面时。

基于事件驱动的程序模型

4.消息

●当Windows捕获一 条事件后 ,它会编写一条消息,将相关信息放入一个数据结构MSG中,然后将包含此数据结构的消息发给需要消息的程序。。

●消息处理是Windows应用程序的核心。

●消息是操作系统通知应用程序某件事情已经发生的一种方式。

●Windows消息是在Windows.h文件中用宏定义的常数。

●消息常数名常以WM_ 开头,格式为WM_XX。

(1)Windows消息来源

Windows应用程序的消息来源于以下四种:

(1)输入消息:键盘和鼠标的输入。此类消息先放在系统消息队列中,然后由Windows将它们送入应用程序消息队列中,由应用程序来处理消息。

(2)控制消息:与Windows的控制对象,如列表框、复选框等进行双向通信。这类消息一般不经过应用程序的消息队列,而是直接发送到控制对象。

(3)系统消息:对程序化的事件或系统时钟中断作出的一些反应。

(4)用户消息:程序员自己定义并在应用程序中主动发生的,一般由应用程序的某一部分内部处理。

VC+ +存在几种系统定义的消息分类,不同的前缀符号经常用于识别不同的消息类,系统定义的消息宏前缀如下:

BM :按钮控制消息

CB :组合框控制消息

DM :默认下压式按钮控制消息

EM :编辑控制消息

LB :列表框控制消息

SBM :滚动条控制消息

WM :窗口控制消息

(2)Windows消息数据结构

消息号:由事先定义好的消息名标识

字参数(wParam) :用于提供消息的附加信息

长字参数(IParam) :用于提供消息的附加信息

Windows消息的数据结构

cpp 复制代码
typedef struct tagMSG 
{
HWND hwnd; //窗口句柄,为nul则可检索所有驻留在消息队交中的消息
UINT message; //消息值,由Windows h头文件中的宏定义标识
WPARAM wParam; //包含有关消息的附加信息,不同消息其值不同
LPARAM IParam;
DWORD time; 1指定消息送至队列的时间
POINT p:/消息发送时屏幕光标位置,POINT是一个结构体
}MSG;

注意:字参与长字参是作为消息的附加信息的,它与具体消息号的值有关;Windows中消息用结构体MSG表示;DWORD是32位无符号整型;

POINT是一个结构体:

cpp 复制代码
typedef struct tagPOINT
{
LONGX;
LONG y; 
}POINT,

二、消息队列和API

1. Windows消息队列

●当产生某消息时,该消息进入消息队列,操作系统根据消息提供的信息值决定由哪个应用程序来处理;

应用程序 按一定的方式查找应用程序中各个类的消息映射(一组宏,用来确定某个消息及相应的处理程序;的对应关系) ;

●找到处理程序后,由处理程序执行相应的操作。

●消息队列是一一个系统定义的内存块,用于临时存储消息,或把消息直接发给窗口函数。

●每个窗口维护自己的消息队列,并从队列中取出消息,利用窗口函数进行处理。

2. Windows程序常用消息

( 1 )标准Windows消息(前缀为WM_ )

分为三类:鼠标消息、键盘消息和窗口消息

(2)控件消息

控件或其他子窗口向父窗口发送WM_ COMMAND消息

(3 )命令消息

菜单项、工具栏按钮、加速键等用户界面对象发送的WM_ COMMAND消息。

●焦点消息: WM_ SETFOCUS,WM_ KILLFOCUS

●定时器消息: WM_ TIMER

( 1 )标准Windows消息- -鼠标消息

-WM_ LBUTTONDOWN :单击鼠标左键时产生的消息

-WM_ LBUTTONUP :放开鼠标左键时产生的消息

-WM_ RBUTTONDOWN ;单击鼠标右键时产生的消息

-WM_ RBUTTONUP :放开鼠标右键时产生的消息

-WM_ LBUTTONDBLCLK :双击鼠标左键时产生的消息

-WM_ RBUTTONDBLCLK :双击鼠标右键时产生的

-WM_ MOUSEMOVE :鼠标在窗C中移动时产生的

( 1 )标准Windows消息- -键盘消息

-WM_KEYDOWN :按下一一个非系统键时产生的消息,附加信息参数wParam为按下键的虚拟键码 ,虚拟键码用以标识按下或释放的键, IParam记录按键的重复次数、扫描码等状态信息;

  • WM_KEYUP :弹起一个非系统键时产生的消息

  • WM_CHAR :按下一个非系统键时产生的消息加信息参数wParam为按下键的ASCII 码,IPare的意思同上;

( 1 )标准Windows消息- -窗口消息

  • WM_CREATE : CreateWindows函数产生的消息;

-WM_CLOSE :关闭窗口时产生的消息;

  • WM_DESTROY :消除窗口时由DestroyWindows()发出的消息

  • WM_QUIT :退出应用程序时由PostQuitMessage()发出的消息

  • WM_ PAINT ; Windows系统需重绘时产生的消息。

  • WM_ SIZE

  • WM_MOVE

3.对象与句柄

●句柄是代表某些资源的一个4字节长的数值,用于标识应用程序中不同的对象和同类对象中不同的实例

●所有的句柄类型以H开头。常见Windows对象的句柄:

4.关于API

●应用程序是如何通知操作系统执行某个功能的呢?

-应用程序要完成某个功能是以函数调用的形式实现;

-同样,应用程序也是以函数调用的方式来通知操作系统执行相应的功能的。

●操作系统把它所能够完成的功能以函数的形式提供给应用程序使用,应用程序对这些函数的调用就叫做系统调用。

●这些函数的集合就是Windows操作系统提供给应用程序编程的接口(Application Programming Interface),简称Windows API。如CreateWindow()函数。

●API是大量函数加上数字常量、宏、结构、类型以及其它相关项的集合。

●大多数API函数定义在Windows.h的文件中,也可以从Windows平台软件开发包( SDK )中得到。

●根据Windows API函数的功能,可以将其分为三类

(1)管理函数 :实现窗口的创建、移动和修改功能

(2)图形设备函数 :实现与设备无关的图形操作功能

(3)系统服务函数:实现与操作系统有关的多种功能

三、Win32程序结构

1、Win32程序结构-WinMain函数

Win32程序示例

例5-1创建一个如图所示的Windows窗口。 本例的目的在于说明创建Windows应用程序的方法、过程及消息的使用。

注意:●LRESULT 选择"右键/ Go To Definition"可查看定义

●对于CALLBACK,表示函数调用方式,stdcall 与 cdecl是两种不同的函数调用约定,区别在函数参数入栈的顺序,由调用函数还是被调用函数将参数弹出栈,以及产生函数修饰名的方法。

1. WinMain函数

功能:

(1)注册窗口类,建立窗口及执行其他必要的初始化工作。

(2)进入消息循环,根据从消息队列中接收的消息,调用相应的处理过程。

(3)当消息循环检索到WM_ QUIT消息时,终止程序拟

WinMain函数有三个基本的组成部分:函数说明、初始化和消息循环。

  1. WinMain函数:函数说明
cpp 复制代码
int WINAPI WinMain
( HINSTANCE hThisInst, ||应用程序当前实例句柄
HINSTANCE hPrevInst,   ||应用程序其他实例句柄
LPSTR IpszCmdLine,     ||指向程序命令行参数的指针
int nCmdShow     ||应用程序开始执行时窗口显示方式的整数值标识,如最大化显示或最小化显示或隐藏显示
)

2)创建一个完整的窗口需要经过下面四个操作步骤:

初始化--窗口类的定义

cpp 复制代码
typedef struct tagWNDCLASS
{
UINT style;    //窗口类的样式,一般设置为0
WNDPROC *IpfnWndProc; //定义指向窗口函数的指针
int cbClsExtra;    //窗口类变量占用的存储空间
int cbWndExtra;    //实例变量占用的存储空间
HINSTANCE hInstance;    //该类的应用程序的实例句柄
HICON hIcon;    //图标,调用Loadlcon
HCURSOR hCursor;    //光标,调用L oadCursor
HBRUSH hbrBackground; //背景刷,调用GetStockObject
LPCTSTR lpszMenuName; //窗口类菜单资源名
LPCTSTR IpszClassName; //窗口类名
}WNDCLASS;

案例的窗口类定义代码:

cpp 复制代码
//窗口类的定义
WNDCL ASS wndclass ;    //窗口类对象
wndclass .style=0;    //窗口类型为缺省类型
wndclass. lpfnWndProc=WndProc; //定 义窗口处理函数名
wndclass.cbClsExtra=0;    //窗口类无扩展
wndclass.cbWndExtra=0;    //窗口实例无扩展
wndclass. hInstance=hInstance; //当前实例句柄
wndclass.hlcon=L oadlcon(NULL,IDI_APPLICATION);    //窗口的最小化图标为缺省图标
wndclass. hCursor=L oadCursor(NULL,IDC_ARROW);    //窗口采用箭头光标 
wndclass. hbrBackground=(HBRUSH)GetStockObject(WHITE_BRUSH); //窗口背景为白色
wndclass.IpszMenuName=NULL; //窗口中无菜单
wndclass. IpszClassName=lpszClassName;//窗类名

Loadlcon:在应用程序中加载一个窗口图标。 原型:HICON Loadlcon(HINSTANCE hInstance, LPCTSTR IplconName )

其中第一个参数(HINSTANCE hInstance)表示图标资源所在的模块句柄,NULL则使用系统预定义图标;第二个参数(LPCTSTR IplconName)表示图标资源名或系统预定义图标标识名

应用程序调用函数GetStockObject获取系统提供的背景刷HBRUSH GetStockObject(int nBrush)

窗口注册代码:

cpp 复制代码
//窗口类的注册
if(!RegisterClass( &wndclass)) //如果注册失败则发出警告
{ MessageBeep(0);
return FALSE;
}

窗口创建:

cpp 复制代码
//创建窗口
HWND hwnd;//窗口句柄
hwnd = CreateWindow
(IpszClassName,        //窗口类名
IpszTitle,         //窗口实例的标题名,
WS_ OVERLAPPEDWINDOW, //窗口的风格
CW_ USEDEFAULT,     //窗口左 上角坐标为缺省值
CW_ USEDEFAULT,
CW USEDEFAULT,    //窗口的高和宽为缺省值
CW USEDEFAULT,
NULL,        //此窗口无父窗口
NULL,        //此窗口 无主菜单
hInstance,     //创建此窗口的应用程序的当前句柄
NULL        //不使用该值
),

创建一个窗口类的实例由函数CreateWindow()实现

cpp 复制代码
HWND CreateWindow
(LPCTSTR IpszClassName, ||窗口类名
LPCTSTR IpszTitle, || 窗口标题名
DWORD dwStyle, || 创建窗口的样式,如下页表所示
int x,y,    ||窗口左上角坐标
int nWidth,nHeight, ||窗口宽度和高度
HWND hwndParent, ||无父窗口则为NULL
HWENU hMenu, ||窗口主菜单句柄
HINSTANCE hInstance, || 创建窗的应用程序当前句柄
LPVOID IpParam ||创建窗口时指定的额外参数
)

常用窗口样式

运用样式方法:#define WS_OVERL APPEDWINDOW(WS OVERL APPED | WS_ CAPTION | WS_ SYSMENUIWS THICKFRAME | WS MINIMIZEBOX | WS MAXIMIZEBOX)

显示窗口:

cpp 复制代码
//显示窗口
ShowWindow( hwnd, nCmdShow);
//立即发送WM_ PAINT消息来刷新窗口客户区
UpdateWindow(hwnd);

消息循环:

cpp 复制代码
//消息循环
MSG Msg;//消息
while( GetMessage(&Msg,NULL,0,0))
{ TranslateMessage( &Msg); //将虚拟键消息转换为字符消息
DispatchMessage( &Msg);    //调度一个消息给窗口程序。通常调度从GetMessage取得的消息
return Msg.wParam; //消息循环结束即程序终止时将信息返回
}

整合的窗口代码:

cpp 复制代码
LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM);
int WINAPI WinMain(HINS TANCE hInstance,HINSTANCE
hPrevInst,LPSTR IpszCmdLine,int nCmdShow)
{ WNDCL ASS wndclass ; //窗口类的定义
 ...           //为窗口类结构体中的成员赋值
RegisterClass( &wndclass));
CreateWindow(...);
ShowWindow( hwnd, nCmdShow);
UpdateWindow(hwnd);
//消息循环
MSG Msg;//消息
while( GetMessage(&Msg,NULL,0,0))
{ TranslateMessage( &Msg); //将虚拟键消息转换为字符消息
DispatchMessage( &Msg);    //调度一个消息给窗口程序。通常调度从GetMessage取得的消息
return Msg.wParam; //消息循环结束即程序终止时将信息返回
}
  1. WinMain函数:消息循环

函数GetMessage():返回零值,即检索到WM_QUIT消息,程序结束循环并退出

cpp 复制代码
GetMessage
( lpMSG, //指向MSG结构的指针 ,
//它包含有从应用程序消息队列中检索到的一条消息数据
hwnd, //指定为哪个窗口检索消息,若为NULL,
//则检索调用该函数的应用程序的所有消息
nMsgFilteMin, /指定检索在Min和Max范围内的消息
nMsgFilterMax
//若这两个参数都为零,该函数检索所有的可用的消息
)

函数TranslateMessage(&Msg):将原始键盘消息WM_KEYDOWN和WM_ KEYUP转化为字符(WM_CHAR)消息

函数DispatchMessage(&Msg):将消息传送到指定窗口函数

2.窗口函数WndProc

WndProc定义了应用程序对接收到的不同消息的响应;它包含了对各种可能接收到的消息的处理过程。

●WndProc函数由一个或多个switch语句组成。

●每一条case语句对应一种消息,当应用程序接收到一个消息时,相应的case语句被激活并执行相应的响应程序模块。

cpp 复制代码
LRESULT CALLBACK WndProc( HWND hwnd, UINT
messgae, WPARAM wParam,LPARAM IParam )
{.
..
switch(message)    || message为标识的消息
{ case..
    ..
    break;
..
    case WM_ DESTROY:
        PostQuitMessage(0); break; 
    default:
        return
DefWindowProc(hwnd,message,wParam,IParam);    //为未定义处理过程的消息提供缺省处理
}
return(0); 
}

在消息处理程序段中一般都有对WM_ DESTROY的处理,该消息是关闭窗口时发出的。它向应用程序发出WM_ QUIT消息,请求退出处理函数:

void PostQuitMessage(int nExitCode) //nExitCode为应用程序的退出代码

案例窗口函数:

cpp 复制代码
// 窗口(消息)处理函数
LRESULT CALL BACK WndProc(HWND hWnd, UINT message, WPARAM wParam, LPARAM IParam)
{
    HDC hdc; // 设备描述表的句柄
    switch (message)
    {
    case WM_PAINT:
        PAINTSTRUCT ps;
        hdc = BeginPaint(hWnd, &ps);
        TextOut(hdc, 0, 0, "武汉理工大学理学院", strlen("武汉理工大学理学院"));
        EndPaint(hWnd, &ps);
        break;
    case WM_CHAR:
        char szChar[20];
        sprintf(szChar, "char %C is %d", wParam, wParam); // 格式输出内容到szChar
        MessageBox(hWnd, szChar, "显示", 0);
        break;
    case WM_LBUTTONDOWN:
        MessageBox(hWnd, "mouse left clicked", "提示", 0);
        hdc = GetDC(hWnd);
        TextOut(hdc, 0, 50, "物理系电信科", strlen("物理系电信科"));
        ReleaseDC(hWnd, hdc);
        break;
    case WM RBUTTONDOWN:
        MessageBox(hWnd, "mouse right clicked", "提示", 0);
        hdc = GetDC(hWnd);
        TextOut(hdc,0, 100, "物理系光信科" ,strlen("物理系光信科");
        ReleaseDC(hWnd, hdc);
    case WM_CLOSE:
        if(IDYES==MessageBox(hWnd, "是否真的结束?", "关闭提示",MB_YESNOCANCEL))
            DestroyWindow(hWnd);
        break;
    case WM_DESTROY:
        PostQuitMessage(0); //调用PostQuitMessage发出WM_ QUIT消息
        break;
    default:    //缺省时采用系统消息缺省处理函数
        return DefWindowProc(hWnd,message,wParam,lParam);
    }
    return (0);
}

Windows程序和消息的基本流程:

相关推荐
Eternal-Student11 分钟前
预处理、编译、汇编、链接
linux·汇编·windows
sp_wxf1 小时前
Stream流
linux·服务器·windows
bcdaren3 小时前
《Windows PE》4.2 绑定导入表
c语言·汇编·windows·pe
芯的一天3 小时前
windows下DockerDesktop命令行方式指定目录安装
windows·docker
localbob4 小时前
uniapp超全user-agent判断 包括微信开发工具 hbuilder mac windows 安卓ios端及本地识别
windows·macos·uni-app·user-agent
科研小达人4 小时前
ChatGPT进行文本分类
windows·chatgpt
yufei-coder13 小时前
掌握 C# 中的 LINQ(语言集成查询)
windows·vscode·c#·visual studio
立秋678916 小时前
Python的defaultdict详解
服务器·windows·python
Indigo_code17 小时前
【数据结构】【链表代码】合并有序链表
数据结构·windows·链表
暮雪倾风17 小时前
【WPF开发】超级详细的“文件选择”(附带示例工程)
windows·wpf