转自: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)项目, 否则生成的程序还带有控制台.

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

【转】C++ win32窗口创建详解相关推荐

  1. C/C++Win32编程基础详解视频下载

    课题视频:C/C++Win32编程基础详解 视频知识:win32窗口的创建 windows事件机制 主讲:择善Uncle老师 学习交流群:386620625 验证码:625 ------------- ...

  2. Django框架学习(一)Django框架安装和项目创建详解

    Django框架学习(一)Django框架安装和项目创建详解 文章目录 Django框架学习(一)Django框架安装和项目创建详解 一.简介 1.1介绍 1.2 URL 1.3.框架原理 二.安装 ...

  3. 站长在线Python精讲:在Python中函数的定义与创建详解

    欢迎你来到站长在线的站长学堂学习Python知识,本文学习的是<在Python中函数的定义与创建详解>.本文的主要内容有:函数的定义.函数的定义规则.函数的创建. 目录 1.函数的定义 2 ...

  4. Windows下游戏制作(1)---win32窗口创建(2)

    写出一个窗口 之前我们简单的认识了一下win32主函数,下面我们来讨论一下如何用这个主函数作为起点来写出一个win32窗口程 序. int WINAPI WinMain(   HINSTANCEhIn ...

  5. win32窗口创建之通俗易懂版

    win32应用程序的第一章 1最简单的win32窗口的创建 这里呢我把win32窗口的创建作为第一个知识分享给大家,也算我对他的一个巩固了.嘿嘿 这里我要说明的是创建这个窗口有两种方法(当然是对于萌新 ...

  6. python怎么打开shell界面-使用IDLE的Python shell窗口实例详解

    启动IDLE后会打开Python shell窗口.当键入代码 时,它会基于Python语法提供自动缩进和代码着色功能. 使用IDLE中的Python shell.代码在输入时会自动着色(基于Pytho ...

  7. linux守护进程原理及创建详解

    在linux或者unix操作系统中在系统的引导的时候会开启很多服务,这些服务就叫做守护进程.为了增加灵活性,root可以选择系统开启的模式,这些模式叫做运行级别,每一种运行级别以一定的方式配置系统. ...

  8. MSG结构体和WndProc窗口过程详解

    MSG结构体和WndProc窗口过程对于Windows编程非常重要,如果不了解它们,可以说就没有学会Windows编程. MSG结构体 MSG 结构体用来表示一条消息,各个字段的含义如下: typed ...

  9. oracle dialog运行,win32窗口创建 之 CreateDialog和DialogBox

    DialogBox 函数原型: INT_PTR DialogBox( HINSTANCE hInstance,LPCTSTR lpTemplate, HWND hWndParent,DLGPROC l ...

最新文章

  1. 前端性能优化 —— 项目瘦身
  2. MVC5学习系列--Razor视图(一)
  3. 【博客搬家旧文】剑指offer [ java ] 面试题10 斐波那契数列
  4. (转) C#如何使用异步编程
  5. 深入理解redis数据类型
  6. spring AbstractBeanDefinition创建bean类型是动态代理类的方式
  7. 解决ubuntu系统root用户下Chrome无法启动问题
  8. python3抓取b站弹幕_python3写爬取B站视频弹幕功能
  9. chage 用户密码管理
  10. C# 学生简单管理系统 数据库 1.0版本
  11. python语言开发平台_Go+Python双语言混合开发
  12. mfc与mysql_MFC与MySql的链接 VS2008
  13. apache+tomcat,搭建负载均衡服务器
  14. 智能机器人机器人心得_如果机器人说到上帝
  15. Iocomp Ultra Pack ActiveX 5.12
  16. 软件开发工程师应该具备哪些证书_初级软件工程师证书简介
  17. ES索引重建reindex详解
  18. 数独游戏代码C++解法
  19. HTML2CANVAS 合成图片
  20. 插入式CAN总线隔离适配器LCAN-Opto在高空作业平台上隔离干扰、调理信号的应用案例介绍

热门文章

  1. django-中间件
  2. Java开启/关闭tomcat服务器
  3. JS取消浏览器文本选中的方法
  4. 将10进制整数转换成16进制整数输出
  5. lua cURL使用笔记
  6. [剑指offer]面试题第[58-2]题[JAVA][左旋转字符串][拼接]
  7. ansible管理mysql安装初始化_[ansible]-ansible初始化mysql数据库
  8. 华为做raid5步骤_华为验厂验厂流程如何?主要内容是什么呢?
  9. 康宁玻璃ct值计算公式_【钦州】CT室铅板生产厂家
  10. 已触发了一个断点 vs_VSCode源码分析-断点调试