【转】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,一经查实,立即删除!

相关文章

2.3 Factory Method(工厂方法)

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

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

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

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

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

【转】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;但…

谈谈基于SQL Server 的Exception Handlingp[下篇]

六、SqlException 在上面一节中&#xff0c;我给出了一个完整的例子说明了&#xff1a;如何在将message定义在sys.messages中保证message的一致性和可维护性&#xff1b;如何在Stored procedure中使用RAISERROR将一个可预知的Error抛出&#xff1b;如何在Stored procedure中使用…

【转】修改static控件背景颜色和文字颜色

当 static 控件或具有 ES_READONLY 风格的 edit 控件被绘制时&#xff0c;会向父窗口发送 WM_CTLCOLORSTATIC 消息。如果我们在窗口过程中处理该消息&#xff0c;就必须返回一个画刷句柄&#xff0c;Windows 会使用该画刷来绘制控件背景&#xff08;子窗口背景&#xff09;。 也…

java中随机数彩票练习_基于javascript实现彩票随机数生成(简单版)

本文实例讲解了JavaScript 实现彩票中随机数组的获取详细代码&#xff0c;分享给大家供大家参考&#xff0c;具体内容如下效果图&#xff1a;具体代码&#xff1a;Math.random方法彩票随机数的生成开始获取随机数组//获取节点var btnGo document.getElementById("btnGo&q…

【转】Windows编程之滚动条—滚动条消息

在用鼠标单击滚动条或者拖动卷动方块时&#xff0c;Windows给窗口消息处理程序发送WM_VSCROLL&#xff08;供上下移动&#xff09;和WM_HSCROLL&#xff08;供左右移动&#xff09;消息。在滚动条上的每个鼠标动作都至少产生两个消息&#xff0c;一条在按下鼠标按钮时产生&…

juc是什么java_JUC简介

JUC是什么JUC是 在Java 5.0添加的 java.util.concurrent包的简称&#xff0c;目的就是为了更好的支持高并发任务&#xff0c;让开发者利用这个包进行的多线程编程时可以有效的减少竞争条件和死锁线程。JUC的结构1&#xff0c;tools(工具类)&#xff1a;又叫信号量三组工具类&am…

WinAPI: SetTextColor - 设置设备环境的文本颜色

//声明: SetTextColor(DC: HDC; {设备环境句柄}Color: COLORREF {颜色值} ): COLORREF; {返回指定的颜色值; 如果返回值是 CLR_INVALID 表示设置失败}//举例: procedure TForm1.Button1Click(Sender: TObject); beginSetTextColor(Canvas.Handle, clRed);Canvas.Te…

java volidate线程安全_03.(多线程与并发)面试题-02--Volidate的原理和指令重排序

线程栈(线程的工作内存)保存了线程运行时候变量值信息。当线程访问某一个对象时候值的时候&#xff0c;首先通过对象的引用找到对应在堆内存的变量的值&#xff0c;然后把堆内存变量的具体值load到线程本地内存中&#xff0c;建立一个变量副本&#xff0c;之后线程就不再和对象…

选址问题java_学习使用分治算法来解决邮局选址问题(Java实现)

邮局选址问题(分治算法)前言提示&#xff1a;在算法的学习过程中我们会遇到各种各样的算法思想&#xff0c;其中最常见的就有分治算法思想&#xff0c;而本文的问题邮局选址问题就是基于分治思想而去实现的一个日常问题提示&#xff1a;以下是本文内容&#xff0c;我将对该问题…

WinAPI: GetCurrentThread、GetCurrentThreadId、GetCurrentProcess、GetCurrentProcessId

{返回当前线程的虚拟句柄} GetCurrentThread: THandle;{返回当前线程 ID} GetCurrentThreadId: DWORD;{返回当前进程的虚拟句柄} GetCurrentProcess: THandle;{返回当前进程 ID} GetCurrentProcessId: DWORD;提示:ID 是系统唯一的标识.所谓虚拟句柄, 就是该句柄只在调用进程的进…

Windows Embedded CE 6.0开发初体验(一)Windows CE概述

这篇文章的目的并不是介绍Windows Embedded CE开发的方方面面&#xff0c;只是用一个初涉嵌入式领域的软件开发者的视角来介绍Windows CE开发中最常用的概念和知识&#xff0c;解决大家会在实际开发中碰到的一些小问题&#xff0c;让大家了解嵌入式开发与应用程序开发&#xff…

Windows Embedded CE 6.0开发初体验(二)CE开发环境

CE开发环境 下面就要让今天的主角闪亮登场了——Digi CC 9P开发板。为了感谢Digi提供的开发板&#xff0c;还是给他们做个广告吧&#xff1a;这是一家以WiFi相关产品为主营业务的公司&#xff0c;也生产ARM架构的CPU&#xff0c;所以做开发板也是他们公司份内的事情了。总的感…

Windows Embedded CE 6.0开发初体验(三)设置Boot-loader

在Digi CC 9P开发板中已经预置了Boot-loader和一个CE 6.0的系统&#xff0c;所以&#xff0c;我们直接启动开发板&#xff0c;就可以运行一个CE 6.0的操作系统。不过&#xff0c;我们的最终目的是要开发一个自己的CE 6.0平台&#xff0c;所以我们需要对Boot-loader中的一些系统…

【转】C++学习三 模板类出错总结(Missing template arguments before ‘L‘)

一、模板类的说明 模板类有一个好处是可以放宽你输入的数据类型。 比如有这样的一个函数&#xff1a; int add(int x, int y) {return xy; } 这个函数对于int类型的x,y才适合&#xff0c;但是如果我们希望计算float类型的呢&#xff1f; 这必须重新定义一个函数&#xff…

Windows Embedded CE 6.0开发初体验(四)跑个应用程序先

因为我们的开发板上已经预置了Windows CE 6.0的系统&#xff0c;所以我们可以先创建一个.NET CF的应用程序&#xff0c;来看一下如何将.NET CF程序部署到Digi开发板上。 第一步&#xff0c;首先在Visual Studio 2005中创建一个新的工程。这一步和开发普通的Windows Mobile应用…

【转】C++学习四 冒泡排序法的一些改进

冒泡排序法需要两次扫描&#xff0c;所以从时间复杂度来说&#xff0c;是O(n2). 如果用图形表示&#xff0c;是这样的&#xff1a; 但是我们可以加以改进。 首先是&#xff0c;如果在排序中间&#xff0c;整个向量已经达到了有序状态&#xff0c;可以直接跳出来。 这样它的复…