【转】C++ win32窗口创建详解

转自:https://my.oschina.net/u/4328928/blog/3315324

本篇所讲解的内容仅限于 Windows 操作系统且限于 win32程序设计

现在我们在Windows系统上用的软件, 早已不是控制台界面, 而是窗体应用程序

窗体与控制台的区别就是: 有了窗口的概念

由于C++的语法复杂, 使得很多人错误地认为C++并不能编写Windows平台的应用程序, 而只能编写由 main 函数为入口的控制台应用程序,这显然是错误。

虽然C++ 编写窗体程序复杂 (这使得很多人放弃学习C++窗体) ,但它具有更强的灵活性, 因为C++代码能够在窗体的任何地方添加想要的操作

C++ 在函数入口上下了很大的功夫, 普遍使用的是 main 函数, 但是如果不是 main 函数为入口的呢?

函数入口的不同导致了生成的结果不同, 下表列出了函数入口所对应生成的结果:

函数入口生成情况
main

控制台应用程序(.exe)

WinMain

窗体应用程序(.exe): 以notepad.exe为例

 

DLLMain

链接库(.lib或.dll)

 

这里我们要讲的是以WinMain函数为入口的win32应用程序

win32 应用程序有以下创建步骤:

1. 向系统注册, 并规定该应用程序的基本信息并绑定事件处理函数

2. 创建窗口(HWND), 并规定该窗口的样式

以上任何一步骤发生错误, 窗口都无法形成

3. 成功后进入消息循环, 将所接收到的消息发送给事件处理函数处理

 

事件 (消息) 处理函数有以下基本内容:

1. 收到消息后进行判断, 并作出相应的数据处理

2. 一旦收到退出的消息(WM_QUIT), 摧毁窗口并结束程序

 

要用到的头文件:

#include <windows.h>

首先, 先认识一下WinMain函数:

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, HINSTANCE, LPWSTR lpCmdLine, int nCmdShow);

关于WinMain的解释 Microsoft 较为详细, 以下内容摘自 Microsoft Docs 的翻译

WinMain是程序入口点。程序启动时,它会注册一些有关应用程序窗口行为的信息

hInstance: 被称为“实例的句柄”或“模块的句柄”。当可执行文件(EXE)加载到内存中时,操作系统使用该值来标识该可执行文件(EXE)。某些Windows功能需要实例句柄,例如,加载图标或位图

hPrevInstance:没有任何意义。它曾在16位Windows中使用,但现在始终为零

lpCmdLine: 包含命令行参数作为Unicode字符串 (初学者可以先跳过)

nCmdShow: 是一个标志,指示是否将主应用程序窗口最小化,最大化或正常显示

 

进入步骤1, 向系统注册

注册要用到 WNDCLASSEX 类型, 其用来存储该程序的基本信息

基本信息:

1. 所给的缓存空间 (包括事件处理函数), 一般为0

wnd.cbClsExtra=0;
wnd.cbWndExtra=0;
wnd.cbSize=sizeof(WNDCLASSEX);

2. 绑定模块句柄(hInstance)与事件处理函数, 菜单栏资源

wnd.hInstance=hInstance;//模块句柄
wnd.lpfnWndProc=WndProc;//事件处理函数(假设为WndProc)
wnd.lpszMenuName=nullptr;//本篇未涉及到菜单栏的讲解, 没有指向任何菜单栏资源, 故为空

3. 设定该程序的绘制风格窗口类名

wnd.style=CS_HREDRAW|CS_VREDRAW;//绘制风格
wnd.lpszClassName="WindowClass";//窗口类名(任意, 但是确定了不能更改),可以理解为该应用程序窗口的"代表"名字

CS_HREDRAW: 窗口发生移动或变化时使宽度变化, 重新绘制窗口

CS_VREDRAW: 窗口发生移动或变化时使高度变化, 重新绘制窗口

没有其中之一可能会使文字的输出在调整窗口后发生错误

4. 设定该窗口的背景色, 图标, 以及鼠标挪到窗口上时的显示样式

wnd.hbrBackground=(HBRUSH)COLOR_WINDOW;//有很多颜色, 这里选用浅灰色, 可以自己到头文件里选
wnd.hIcon=LoadIcon(hInstance,IDI_APPLICATION);//默认的图标
wnd.hIconSm=LoadIcon(hInstance,IDI_APPLICATION);//指向图标资源的指针, 这里采用默认图标
wnd.hCursor=LoadCursor(hInstance,IDC_ARROW);//标准的鼠标样式

有多种样式, 自己查, 不做过多介绍

合写为:

WNDCLASSEX wnd;
wnd.hbrBackground=(HBRUSH)COLOR_WINDOW;
wnd.lpfnWndProc=WndProc;
wnd.style=CS_HREDRAW|CS_VREDRAW;
wnd.lpszClassName="WindowClass";
wnd.hInstance=hInstance;
wnd.hIcon=LoadIcon(hInstance,IDI_APPLICATION);
wnd.hIconSm=LoadIcon(hInstance,IDI_APPLICATION);
wnd.hCursor=LoadCursor(hInstance,IDC_ARROW);
wnd.cbSize=sizeof(WNDCLASSEX);
wnd.lpszMenuName=nullptr;
wnd.cbClsExtra=0;
wnd.cbWndExtra=0;

现在注册并检查是否成功, 用到函数 RegisterClassEx ,向系统提交注册信息, 失败返回0

if(!RegisterClassEx(&wnd)) //注册并检查是否失败
{MessageBox(nullptr,"Error!","Error",MB_ICONERROR|MB_OK);return 0; //失败后结束程序
}

 

 

 

执行步骤2, 创建窗口并规定样式:

认识创建窗口的函数(参考自百度百科 ):

HWND CreateWindowEx(
DWORD DdwExStyle,        //窗口的扩展风格
LPCTSTR lpClassName,    //指向注册类名的指针
LPCTSTR lpWindowName,   //指向窗口名称的指针
DWORD dwStyle,          //窗口基本样式
int x,                  //窗口的一开始出现在电脑屏幕上的x坐标
int y,                  //窗口的一开始出现在电脑屏幕上的y坐标int nWidth,             //窗口的宽度
int nHeight,            //窗口的高度
HWND hWndParent,        //父窗口
HMENU hMenu,            //菜单栏
HINSTANCE hInstance,    //该应用程序的模块句柄
LPVOID lpParam          //指向窗口的创建数据
);

这里窗口的扩展, 基本风格有很多, 不多介绍, 以标准的窗口风格为例

这里选择几个难懂的参数做解释:

DdwExStyle: 这里采用标准扩展风格 WS_EX_CLIENTEDGE

lpClassName: 这里可理解为窗口所"代表的名字", 上面注册时已编写, 即"WindowClass"

lpWindowName: 这里可以理解为窗口的标题

dwStyle: 这里采用标准窗口风格 WS_OVERLAPPEDWINDOW

hMenu: 该窗口所包含的菜单栏, 这里没有创建菜单栏, 即为 nullptr

hInstance: 直接将WinMain中的 hInstance 填入

lpParam: 不常用, 直接写 nullptr

 

创建完了, 不代表能够显示出来, 所以还要显示窗口

BOOL ShowWindow(HWND hWnd, int nCmdShow); //显示窗口的函数

hWnd: 要显示的窗口

nCmdShow: 窗口的显示方式, WinMain 参数中已有该窗口的显示方式, 直接将参数 nCmdShow 写入

 

上代码:

HWND hwnd=CreateWindow("WindowClass","Title",WS_OVERLAPPEDWINDOW,0,0,800,800,nullptr,hMenu,hInstance,nullptr);
ShowWindow(hwnd,nCmdShow);

这是一个出现在(0,0)左上角, 800px * 800px, 标题栏显示 Title 的标准父窗口

 

 

 

现在执行步骤3, 进入消息循环

消息循环是这样的:

       在窗体程序创建并显示成功后, 需要进行各种各样的消息的处理 (如, 鼠标按下了窗口中的按钮, 键盘按下了一个按键等), 这些信息就是消息(MSG), 这些消息要被程序所捕获,用不断循环来检查是否存在消息, 将消息转换为可以被事件处理函数所接受的内容并发送给事件处理函数.

 

而如何捕获消息呢? 每个应用程序创建后, 系统都会为其开辟一个"消息队列", 将所接收到的各种消息入队:

 

 

 而程序要做的就是不断从队首元素中获取消息直到队首元素为空(即收到了WM_QUIT的退出消息)为止, 结束程序

消息的表示: WM_+ 消息内容

如:

退出消息: WM_QUIT

创建窗口消息: WM_CREATE

调整窗口大小的消息: WM_SIZE

 

上代码 (实现简单): 

MSG msg;
while(GetMessage(&msg,0,0,0)) //不断获取队首元素直到队首为空
{TranslateMessage(&msg); //将收到的消息转换为"WM_+消息内容"的形式DispatchMessage(&msg); //将收到的消息发送给事件处理函数
}

 

 

 

WinMain 函数的部分已完成, 上完整代码:

BOOL RegisterWindow(HINSTANCE hInstance)
{WNDCLASSEX wnd;wnd.hbrBackground=(HBRUSH)COLOR_WINDOW;wnd.lpfnWndProc=WndProc;wnd.style=CS_HREDRAW|CS_VREDRAW;wnd.lpszClassName="WindowClass";wnd.hInstance=hInstance;wnd.hIcon=LoadIcon(hInstance,IDI_APPLICATION);wnd.hIconSm=LoadIcon(hInstance,IDI_APPLICATION);wnd.hCursor=LoadCursor(hInstance,IDC_ARROW);wnd.cbSize=sizeof(WNDCLASSEX);wnd.lpszMenuName=nullptr;wnd.cbClsExtra=0;wnd.cbWndExtra=0;return RegisterClassEx(&wnd);
}
BOOL DisplayWindow(HWND& hwnd,HINSTANCE hInstance,HMENU hMenu,int nCmdShow)
{hwnd=CreateWindow("WindowClass","TweeChaice",WS_OVERLAPPEDWINDOW,0,0,800,800,nullptr,nullptr,hInstance,nullptr);ShowWindow(hwnd,nCmdShow);return hwnd?TRUE:FALSE;
}HWND hwnd;
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPSTR lpCmdLine,int nCmdShow)
{if(!RegisterWindow(hInstance)){MessageBox(nullptr,"Error register!","Error",MB_ICONERROR|MB_OK);return -1;}DisplayMenu();if(!DisplayWindow(hwnd,hInstance,nullptr,nCmdShow)){MessageBox(nullptr,"Error creating window!","Error",MB_ICONERROR|MB_OK);return -1;}MSG msg;while(GetMessage(&msg,0,0,0)){TranslateMessage(&msg);DispatchMessage(&msg);}return 0;
}

 

 

 

现在就差事件处理了, 什么是事件处理函数?

  对一系列的 "WM_消息" 进行对应的处理 (如, 点击按钮后进行处理) 的函数, 它的编写风格很随意, 凭自己选择

 

事件处理函数的返回值必须为: LRESULT CALLBACK

所包含的参数类型必须为如下, 但变量名随意:

LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)

解释:

hwnd: 父窗口

message: 收到的 "WM" 消息

wParam与lParam: MSG中附加消息, 与窗口的信息, 事件处理有关

 

编写随意(带必须拥有WM_QUIT, 否则窗体的关闭将无法进行):

LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{if(message==WM_DESTROY) //当窗口被摧毁时(有时是强制摧毁), 发送WM_QUIT消息, 可以省略PostQuitMessage(0);else if(message==WM_QUIT) //接收到退出消息, 摧毁窗口(并停止接受任何消息, 消息队列中的队首元素清除为空), 然后消息循环将发现队首为空, 实现退出程序的效果DestroyWindow(hwnd);return DefWindowProc(hwnd,message,wParam,lParam); //正常返回
}

 

上完整代码:

#include <windows.h>
LRESULT CALLBACK WndProc(HWND hwnd,UINT message,WPARAM wParam,LPARAM lParam)
{if(message==WM_DESTROY)PostQuitMessage(0);else if(message==WM_QUIT)DestroyWindow(hwnd);return DefWindowProc(hwnd,message,wParam,lParam);
}
BOOL RegisterWindow(HINSTANCE hInstance)
{WNDCLASSEX wnd;wnd.hbrBackground=(HBRUSH)COLOR_WINDOW;wnd.lpfnWndProc=WndProc;wnd.style=CS_HREDRAW|CS_VREDRAW;wnd.lpszClassName="WindowClass";wnd.hInstance=hInstance;wnd.hIcon=LoadIcon(hInstance,IDI_APPLICATION);wnd.hIconSm=LoadIcon(hInstance,IDI_APPLICATION);wnd.hCursor=LoadCursor(hInstance,IDC_ARROW);wnd.cbSize=sizeof(WNDCLASSEX);wnd.lpszMenuName=nullptr;wnd.cbClsExtra=0;wnd.cbWndExtra=0;return RegisterClassEx(&wnd);
}
BOOL DisplayWindow(HWND& hwnd,HINSTANCE hInstance,HMENU hMenu,int nCmdShow)
{hwnd=CreateWindow("WindowClass","TweeChaice",WS_OVERLAPPEDWINDOW,0,0,800,800,nullptr,nullptr,hInstance,nullptr);ShowWindow(hwnd,nCmdShow);return hwnd?TRUE:FALSE;
}HWND hwnd;
int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPSTR lpCmdLine,int nCmdShow)
{if(!RegisterWindow(hInstance)){MessageBox(nullptr,"Error register!","Error",MB_ICONERROR|MB_OK);return -1;}if(!DisplayWindow(hwnd,hInstance,nullptr,nCmdShow)){MessageBox(nullptr,"Error creating window!","Error",MB_ICONERROR|MB_OK);return -1;}MSG msg;while(GetMessage(&msg,0,0,0)){TranslateMessage(&msg);DispatchMessage(&msg);}return 0;
}

执行效果:

 

 win32 编程复杂但不难, 需要好好领悟其原理, 才能走得更远

另: 窗体应用程序不能在单个文件中在生成结果没被设定为GUI 应用程序(窗体应用程序或WIN32)中正确编写, 需创建窗体程序(WIN32)项目, 否则生成的程序还带有控制台.

      窗体应用程序中无法使用任何控制台的输入输出

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/436277.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java 子类继承父类_关于Java 的继承问题,子类会继承父类的哪些东西?-----转载...

和C类似&#xff0c;可以继承基类的公共属性和方法。在Java继承里&#xff0c;父类的属性还有方法在声明时&#xff0c;如果是public关键字即公共属性&#xff0c;则在子类继承时&#xff0c;这些属性和方法都会被子类继承。受保护的也可以继承但是私有的类属性成员和方法则无法…

天气很冷晚饭很香

今天和zhoujia 去吃她家门口的东北菜&#xff0c;席间谈笑风生&#xff0c;突然她说“亲爱的&#xff0c;眼光要看的长远一些”&#xff0c;附带标准的领导手势&#xff0c;让我笑翻。zhoujia是个乐观的人&#xff0c;从来没有见她发愁过&#xff0c;她的房子装修的也差不多了&…

【转】win32窗口的大小,居中,拖动

参考 https://www.cnblogs.com/findumars/p/3948315.html 不让调整大小 窗口风格设置&#xff0c;后面两个是最大最小化按钮 dwStyle^WS_THICKFRAME^WS_MAXIMIZEBOX^WS_MINIMIZEBOX 不让拖动 消息循环里添加 case WM_NCLBUTTONDOWN: { switch (wParam) …

java +号变空格_base64码通过http传输 +号变 空格 问题解决

通过七牛云base64上传图片&#xff0c;通过官方示例上传成功后&#xff0c;根据示例改了一个controller。通过前端往后端传base64码形式进行测试。死活不通过&#xff0c;七牛报400。仔细排查后发现&#xff0c;示例转换的base64码与前端传来的base64码稍有区别&#xff0c;前端…

Linux编程练习 --进程间通信1--无名管道

进程间通信系列--管道 管道可以说是最古老的IPC形式&#xff0c;所谓管道&#xff0c;是指进程间建立的一条通信的通道&#xff0c;从本质上看&#xff0c;管道是UNIX文件概念的推广管道通信的介质是文件&#xff0c;先看一下管道的特点&#xff1a; 1.管道的特点&#xff1a; …

2.3 Factory Method(工厂方法)

【返回目录】 我们现在把场景从险象丛生的特工战切换到更为壮观的二战中来&#xff0c;很多人都看过《拯救大兵瑞恩》&#xff0c;这部电影最著名的两场战斗非首和尾莫属&#xff0c;我们就用最后一场战役来举例说明什么是工厂方法吧。 相比诺曼底登陆&#xff0c;最后一场战役…

【转】CreateWindow函数详解

CreateWindow函数详解 在注册完窗口类后就需要进行窗口的创建&#xff0c;用到的函数理所当然就是CreateWindow()&#xff0c; 而这个函数是基于窗口类的&#xff0c;所以还需要指定几个参数来制定特定的窗口。而且像一些不带边框的窗口是怎么创建的也是具有相当的技巧的&#…

java中处理打折率_【JAVA300例】13、输入价格判断折扣,switch用法+int留整数方便判断...

import java.util.Scanner;public class Test013{public static void main(String[] args){Scanner in new Scanner(System.in);System.out.println("请输入金额(整数):");float money in.nextFloat();String zhekou "";if (money>1200){int grade …

[转] UML中关联、依赖、聚集等关系的异同

1. 关联&#xff1a;连接模型元素及链接实例&#xff0c;用一条实线来表示&#xff1b;2. 依赖&#xff1a;表示一个元素以某种方式依赖于另一个元素&#xff0c;用一条虚线加箭头来表示&#xff1b;3. 聚集&#xff1a;表示整体与部分的关系&#xff0c;用一条实线加空心菱…

【转】C++ _T()函数和_L()函数介绍

_T("")是一个宏,他的作用是让你的程序支持Unicode编码&#xff0c;因为Windows使用两种字符集ANSI和UNICODE&#xff0c; 前者就是通常使用的单字节方式&#xff0c;但这种方式处理象中文这样的双字节字符不方便&#xff0c;容易出现半个汉字的情况。 而后者是双字节…

Linux编程练习 --进程间通信2--两个管道实现双向通信

利用两个管道进行进程间双向通信在第一篇练习已经大致作出说明&#xff0c;下面将进行一个更为综合的练习 首先看题目&#xff1a; 设有二元函数f(x,y) f(x) f(y) 其中&#xff1a; f(x) f(x-1) * x (x >1) f(x)1 (x1) f(y) f(y-1) f(y-2) (y> 2) f(y)1 (y1,2) 请编…

Windows集群网络配置最佳做法

要开始做集群了&#xff0c;找点资料来看看&#xff1a;阅读提示&#xff1a;本文为 Microsoft Windows 2000 或 Windows Server 2003 服务器群集的网络基础结构提供了服务器群集要求和最佳做法。若要群集可以正常运行&#xff0c;必须满足这些要求。最佳做法是从部署反馈和现场…

JOJ的2042面试题目的数学推导过程

JOJ的2042题目是一个程序理解题目&#xff0c;这个题目非常有意思&#xff0c;给出了下面一段C源代码&#xff0c;要求计算出最后的输出结果&#xff0c;源代码如下&#xff1a; #include<cstdio> int main(void) { int x 987654321, c 0, d 1, e 6; while…

linux mysql密码转义_linux忘记mysql密码处理方法

linux忘记mysql密码处理方法:# /etc/init.d/mysql stop# mysqld_safe --usermysql --skip-grant-tables --skip-networking &# mysql -u root mysqlmysql> update user set passwordpassword(newpassword) where userroot;mysql> flush privileges;mysql> quit# /…

不同职业的面试着装技巧。

美女姜培琳&#xff1a;传授独家心经 不同职业的面试着装技巧。 一般来说&#xff0c;职场中精英女性的装扮&#xff0c;首要应讲求端庄、稳重。人们对服饰过于花哨怪异者的工作能力、工作作风、敬业精神、生活态度等&#xff0c;都会持怀疑的态度。 其实&#x…

ARMV4,ARMV4T,ARMV4I的意义

ARMV4,ARMV4T,ARMV4I 以上表示的是指令的版本 也就是三种指令集&#xff1a; ARMv4 -> 它只支持 32 位 ARMv4 指令 ARMv4T ->“T”代表 Thumb(16 位指令模式) ARMv4I ->“I”代表交互作用 (Interworking)。它允许 32 位指令和 16 位指令共存 一些经常出现的CPU支持的指…

【转】C++中的字符串(String)和数值转换

『写在前边』 因为写PAT总是遇到字符串&#xff08;主要是String&#xff09;和数值之间的转化&#xff0c;所以整理一下以便好复习。 『String类型的字符串』 这里数值和字符串相互转换的例子&#xff0c;使用的是stringstream函数&#xff0c;在c11当中有定义好的现成的函数…

好开心

好开心&#xff01;收到我们家lily的礼物&#xff01; 前天&#xff0c;lily传给我一个flash相册&#xff0c;他将我们认识以来的合照做成了flash&#xff0c;配上了音乐。看了好感动&#xff0c;好开心&#xff01; 其实照片倒不是很多&#xff0c;也就7到8张吧&#xff0c;但…

java 实现真正的随机数_关于java:SecureRandom的Android实现是否产生真正的随机数?...

我已经阅读过&#xff0c;一般来说&#xff0c;SecureRandom的一些实现可能会产生真正的随机数。特别是&#xff0c;Android文档说instances of this class will generate an initial seed using an internal entropy source, such as /dev/urandom号但这是否意味着它将产生真正…

etherpeek nx在网络维护中的应用

摘要&#xff1a;该文分析了etherpeek nx的工作原理&#xff0c;主要利用etherpeek nx的数据包截取和分析功能来对netrobocop&#xff08;网络执法官&#xff09;数据包的分析&#xff0c;了解局域网和netrobocop的基本原理&#xff0c;从而更好地维护网络的安全和畅通。 关键…