2024年2月6日发(作者:)

VS2010/MFC编程入门教程之目录和总结

分类标签: 编程入门 VS2010 VC++ MFC

目前该教程可以到鸡啄米编程课堂去学习,阅读体验更好,更适合在线学习。

鸡啄米的这套VS2010/MFC编程入门教程到此就全部完成了,虽然有些内容还未涉及到,但帮助大家进行VS2010/MFC的入门学习业已足够。以此教程的知识为基础,学习VS2010/MFC较为深入的内容已非难事。作为本教程的最后一课,鸡啄米将对前面所讲内容进行目录归纳,并对这八个月加班加点的努力进行总结。

一、VS2010/MFC编程入门教程之目录

第一部分:VS2010/MFC开发环境

VS2010/MFC编程入门之前言

VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)

第二部分:VS2010/MFC应用程序框架

VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)

VS2010/MFC编程入门之三(VS2010应用程序工程中文件的组成结构)

VS2010/MFC编程入门之四(MFC应用程序框架分析)

VS2010/MFC编程入门之五(MFC消息映射机制概述)

第三部分:对话框

VS2010/MFC编程入门之六(对话框:创建对话框模板和修改对话框属性)

VS2010/MFC编程入门之七(对话框:为对话框添加控件)

VS2010/MFC编程入门之八(对话框:创建对话框类和添加控件变量)

VS2010/MFC编程入门之九(对话框:为控件添加消息处理函数)

VS2010/MFC编程入门之十(对话框:设置对话框控件的Tab顺序)

VS2010/MFC编程入门之十一(对话框:模态对话框及其弹出过程)

VS2010/MFC编程入门之十二(对话框:非模态对话框的创建及显示)

VS2010/MFC编程入门之十三(对话框:属性页对话框及相关类的介绍)

VS2010/MFC编程入门之十四(对话框:向导对话框的创建及显示)

VS2010/MFC编程入门之十五(对话框:一般属性页对话框的创建及显示)

VS2010/MFC编程入门之十六(对话框:消息对话框)

VS2010/MFC编程入门之十七(对话框:文件对话框)

VS2010/MFC编程入门之十八(对话框:字体对话框)

VS2010/MFC编程入门之十九(对话框:颜色对话框)

第四部分:常用控件

VS2010/MFC编程入门之二十(常用控件:静态文本框)

VS2010/MFC编程入门之二十一(常用控件:编辑框Edit Control)

VS2010/MFC编程入门之二十二(常用控件:按钮控件Button、Radio Button和Check Box)

VS2010/MFC编程入门之二十三(常用控件:按钮控件的编程实例)

VS2010/MFC编程入门之二十四(常用控件:列表框控件ListBox)

VS2010/MFC编程入门之二十五(常用控件:组合框控件Combo Box)

VS2010/MFC编程入门之二十六(常用控件:滚动条控件Scroll Bar)

VS2010/MFC编程入门之二十七(常用控件:图片控件Picture Control)

VS2010/MFC编程入门之二十八(常用控件:列表视图控件List Control 上)

VS2010/MFC编程入门之二十九(常用控件:列表视图控件List Control 下)

VS2010/MFC编程入门之三十(常用控件:树形控件Tree Control 上)

VS2010/MFC编程入门之三十一(常用控件:树形控件Tree Control 下)

VS2010/MFC编程入门之三十二(常用控件:标签控件Tab Control 上)

VS2010/MFC编程入门之三十三(常用控件:标签控件Tab Control 下)

第五部分:菜单、工具栏与状态栏

VS2010/MFC编程入门之三十四(菜单:VS2010菜单资源详解)

VS2010/MFC编程入门之三十五(菜单:菜单及CMenu类的使用)

VS2010/MFC编程入门之三十六(工具栏:工具栏资源及CToolBar类)

VS2010/MFC编程入门之三十七(工具栏:工具栏的创建、停靠与使用)

VS2010/MFC编程入门之三十八(状态栏的使用详解)

第六部分:文档、视图和框架

VS2010/MFC编程入门之三十九(文档、视图和框架:概述)

VS2010/MFC编程入门之四十(文档、视图和框架:各对象之间的关系)

VS2010/MFC编程入门之四十一(文档、视图和框架:分割窗口)

第七部分:MFC常用类

VS2010/MFC编程入门之四十二(MFC常用类:CString类)

VS2010/MFC编程入门之四十三(MFC常用类:CTime类和CTimeSpan类)

VS2010/MFC编程入门之四十四(MFC常用类:定时器Timer)

VS2010/MFC编程入门之四十五(MFC常用类:CFile文件操作类)

VS2010/MFC编程入门之四十六(MFC常用类:MFC异常处理)

第八部分:字体和文本输出

VS2010/MFC编程入门之四十七(字体和文本输出:CFont字体类)

VS2010/MFC编程入门之四十八(字体和文本输出:文本输出)

第九部分:图形图像

VS2010/MFC编程入门之四十九(图形图像:CDC类及其屏幕绘图函数)

VS2010/MFC编程入门之五十(图形图像:GDI对象之画笔CPen)

VS2010/MFC编程入门之五十一(图形图像:GDI对象之画刷CBrush)

第十部分:Ribbon界面开发

VS2010/MFC编程入门之五十二(Ribbon界面开发:创建Ribbon样式的应用程序框架)

VS2010/MFC编程入门之五十三(Ribbon界面开发:为Ribbon Bar添加控件)

VS2010/MFC编程入门之五十四(Ribbon界面开发:使用更多控件并为控件添加消息处理函数)

二、VS2010/MFC编程入门教程之总结

写这套VS2010/MFC编程入门教程历时八个月,时间比较长,都是鸡啄米在工作之余加班加点完成的。虽然有点累,但看到有这么多人过来支持鸡啄米,很是值得。

写每节教程我都力求讲明白,深入浅出的阐述每个知识点;每个实例都以让大家快速的掌握要点为目的,且都是鸡啄米事先写好并能正常运行的。

每节课的写作耗时差不多都在两三个小时以上,有时候甚至写到凌晨一点多。付出了很多,同样也收获了很多,最大的收获莫过于大家的肯定和支持,很多朋友都留言或者发邮件支持鸡啄米,这说明我的辛勤劳动没有白费,帮助了很多人,感谢那些一直以来支关注鸡啄米的朋友,也特别感谢那些在鸡啄米经常留言支持的读者。

除了大家的支持,鸡啄米自己也通过本教程的完成更深入的理解了VS2010/MFC的很多内容,提高了对VS2010/MFC的认识水平,尤其是更加巩固了一些较少使用的知识。在帮人的同时也帮了自己。

很多朋友在鸡啄米留言或者给鸡啄米发电子邮件讨论问题,由于时间确实比较少,晚上更新文章,白天上班,所以只回复了一部分,望大家见谅。

学会了VS2010/MFC编程,很多人就开始喜欢研究MFC了,但是提醒大家不要忘了根本,即C++语言。从《鸡啄米C++编程入门系列》和《VS2010/MFC编程入门教程》这两套学习教程的受欢迎程度,鸡啄米感觉大家有些忽视C++的学习,更喜欢应用性强的MFC。在你能熟练使用MFC后,最好再认真学学C++,提高C++水平才能真正让你的编程功力上一个台阶。

以后鸡啄米会继续分享各种编程知识,还会编写一些教程,希望大家能一如既往的关注鸡啄米网站,支持鸡啄米!

首页 » 软件开发 » VS2010/MFC编程入门之前言

VS2010/MFC编程入门之前言

分类标签: 编程入门 VC++ VS2010 MFC

鸡啄米的C++编程入门系列给大家讲了C++的编程入门知识,大家对C++语言在语法和设计思想上应该有了一定的了解了。但是教程中讲的例子只是一个个简单的例程,并没有可视化窗口。鸡啄米在这套VS2010/MFC编程入门教程中将会给大家讲解怎样使用VS2010进行可视化编程,也就是基于窗口的程序。

C++编程入门系列主要偏重于理论方面的知识,目的是让大家打好底子,练好内功,在使用VC++编程时不至于丈二和尚摸不着头脑。本套教程也会涉及到VC++的原理性的东西,同样更重视实用性,让大家学完本套教程以后,基本的界面程序都能很容易编写出来。

VC++简介

VC++全称是Visual C++,是由微软提供的C++开发工具,它与C++的根本区别就在于,C++是语言,而VC++是用C++语言编写程序的工具平台。VC++不仅是一个编译器更是一个集成开发环境,包括编辑器、调试器和编译器等,一般它包含在Visual Studio中。Visual Studio包含了VB、VC++、C#等编译环境。当然我们在使用VC++ 6.0的时候为了轻便,总是只单独安装VC++ 6.0。但自微软2002年发布Visual 以来,微软建立了在.NET框架上的代码托管机制,一个项目可以支持多种语言开发的组件,VC++同样被扩展为支持代码托管机制的开发环境,所以.NET Framework是必须的,也就不再有VC++的独立安装程序,不过可以在安装Visual Studio时只选择VC++进行安装。

VC++版本的选择:VS2010

因为VC++ 6.0以后的版本不再有独立的安装程序,所以鸡啄米在教程中将不会称VC++ 6.0以后的版本为VC++ 7.0等等,而是用VC++所属的Visual Studio的版本名称代替,比如VS2003。

近些年VC++主要的版本包括:VC++ 6.0、VS2003、VS2005、VS2008和VS2010。

VC++ 6.0占用的系统资源比较少,打开工程、编译运行都比较快,所以赢得很多软件开发者的青睐。但因为它先于C++标准推出,所以对C++标准的支持不太好。举个例子:

for(int i=0; i<5; i++)

{

a[i] = i;

}

for语句中声明的变量i,对于VC++ 6.0来说,出了for循环仍能使用。但很显然这与C++标准对于变量生存期的规定不符合。

随着VC++版本的更新,对C++标准的支持越来越好,对各种技术的支持也越来越完善。但同时新版本所需的资源也越来越多,对处理器和内存的要求越来越高。到VS2010,光安装文件就2G多,安装后的文件占3G多空间,其运行也经常受处理器和内存等性能的限制。但鸡啄米还是推荐大家使用VS2010,毕竟它是最新版本,类库和开发技术都是最完善的,本教程也将使用VS2010为大家做例程的演示。当然如果系统配置确实比较低,可以选择VS2005,VS2005和VS2010相比还是要轻量级一些的。VC++ 6.0已经过时,奉劝大家尽量别用了。

VC++与MFC

讲VC++免不了要提MFC,MFC全称Microsoft Foundation Classes,也就是微软基础类库。它是VC++的核心,是C++与Windows API的结合,很彻底的用C++封装了Windows SDK(Software Development Kit,软件开发工具包)中的结构和功能,还提供了一个应用程序框架,此应用程序框架为软件开发者完成了一些例行化的工作,比如各种窗口、工具栏、菜单的生成和管理等,不需要开发者再去解决那些很复杂很乏味的难题,比如每个窗口都要使用Windows API注册、生成与管理。这样就大大减少了软件开发者的工作量,提高了开发效率。

当然VC++不是只能够创建MFC应用程序,同样也能够进行Windows SDK编程,但是那样的话就舍弃了VC++的核心,放弃了VC++最强大的部分。MFC也不是只能用于VC++中,它同样也可以用在Borland C++等编译器中,当然没有几个人这样做。

本节旨在让大家对VC++、VS2010和MFC有基本的概念上的认识,后面鸡啄米会带大家进入VS2010/MFC的世界,让大家轻松的开发各种包含窗口、图形等的可视化程序。

首页 » 软件开发 » VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)

VS2010/MFC编程入门之一(VS2010与MSDN安装过程图解)

分类标签: 编程入门 VC++ VS2010

上一讲中鸡啄米对VC++和MFC做了一些简单介绍。在本套教程中鸡啄米将使用VS2010为大家讲解如何使用VC++和MFC进行编程,所以首先要安装VS2010。

一.下载VS2010

首先我们需要下载VS2010,大家可以在网上下载VS2010破解正式版,建议选择英文版,养成使用英文工具的习惯。鸡啄米使用VS2010旗舰试用版为例介绍安装过程,旗

舰试用版官方下载地址为:/download/en/?displaylang=en&id=12187。正式版的安装过程与试用版类似。

二.安装VS2010

下载后进行安装。安装方法与一般的iso文件一样,可以使用虚拟光驱软件Daemon Tools安装,也可以将其解压后点击进行安装。

鸡啄米为了让大家更直观的看到安装过程,我将在自己机子上再重新安装一次,并截图为大家讲解。

这里使用Daemon Tools安装VS2010。首先打开Daemon Tools,屏幕右下角会出现托盘图标,在图标上点右键,会弹出菜单,再把鼠标移到菜单项“虚拟设备”上,然后再移到子菜单项“设备 0:[L:] 无媒体”上,最后点击下一级子菜单项“装载映像”,弹出对话框选择文件。

这样虚拟光驱就会打开此iso文件,弹出自动安装的提示,选择“运行”就可以了,如果没有弹出提示就通过资源管理器进入虚拟光驱,用安装。接着会弹出下面的对话框:

当然选择“Install Microsoft Visual Studio 2010”进入下一步,加载安装组件后如下显示:

点“Next”后:

选择“I have read and accept the license terms”后点“Next”弹出对话框:

此处是让我们选择要安装的功能,有两种:Full(完全)和Custom(自定义)。Full选项表示安装所有编程语言和工具,Custom选择表示可以自定义要安装的编程语言和工具。右侧可以更改安装路径,鸡啄米建议不要安装到C盘,因为它占用的空间比较大。鸡啄米安装到了D盘,使用Full完全安装。如果选择Custom安装,点“Next”则出现如下画面:

大家可以根据自己的需要取消某些语言或工具的安装,比如不想安装Visual C#,取消选择它就可以了。如果觉得以后都有可能会用到,那就像鸡啄米一样选择完全安装吧。

Full或Custom方式和安装路径设置好后,点“Install”进行安装:

可能正式版的安装文件在安装过程中会有重启过程。鸡啄米使用的试用版中间并没有重启。安装完成:

如果要继续安装MSDN,先不要卸载虚拟光驱映像。

三.安装MSDN

我们使用VS2010进行软件开发同样离不开帮助文档,即MSDN。在本地安装MSDN的方法如下:

在开始菜单的“所有程序”->“Microsoft Visual Studio 2010”->“Visual Studio Tools”下选择“Manage Help Settings - ENU”:

弹出对话框:

可以将帮助库存在默认路径,也可以修改存放路径。鸡啄米使用默认路径,点“OK”出现:

选择“Install Content From Disk”后弹出对话框选择帮助所在文件,这时需要在加载了VS2010的虚拟光驱中找,选择图中所示路径:

点OK后出现如下对话框,可以点“Add”选择要添加的帮助库,鸡啄米全部添加了。

点“Update”进行安装,等待其完成就可以了。

使用MSDN时点击开始菜单的“所有程序”->“Microsoft Visual Studio 2010”->“Microsoft Visual Studio 2010 Documentation”即可。

到此VS2010和MSDN的安装过程就结束了。以后就可以正式使用VS2010进行软件开发了。至于VS2010的使用方法在鸡啄米的C++编程入门系列中已经介绍过,大家可以看看。

第二部分:VS2010/MFC应用程序框架

VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)

分类标签: 编程入门 VC++ MFC VS2010

上一讲中讲了VS2010和MSDN如何安装,相信大家都已经安装好了。这一讲给大家一个简单的例子,演示如何生成单文档应用程序框架。

解决方案与工程

鸡啄米在VS2010的使用介绍中已经讲了解决方案与工程的概念,这里再重提一下。每个应用程序都作为一个工程来处理,它包含了头文件、源文件和资源文件等,这些文件通过工程集中管理。在VS2010中,工程都是在解决方案管理之下的。一个解决方案可以管理多个工程,可以把解决方案理解为多个有关系或者没有关系的工程的集合。VS2010提供了一个Solution Explorer解决方案浏览器视图,可以显示当前解决方案的内容,当新建一个工程时可以选择新建一个解决方案还是加入当前解决方案。

下图左侧面板中正在显示的视图就是Solution Explorer,视图中有一个解决方案-HelloWorld,此解决方案下有一个同名的工程-HelloWorld。

在应用程序向导生成应用程序后,VS2010会在用户设置的路径下,以解决方案名为名称建立一个目录,里面存放自动生成的文件。

使用VS2010应用程序向导生成单文档应用程序框架

鸡啄米这里简略演示下怎样生成单文档应用程序框架,让大家先有个直观的了解,有不理解的地方可以留着以后回来再看。下面按照操作步骤一步步讲解:

1.点菜单栏File->New->Project,弹出New Project对话框,我们可以选择工程类型。

如果安装完VS2010以后第一启动时已经设置为VC++,则Installed Templates->Visual C++项会默认展开,而如果没有设置VC++,则可以展开到Installed Templates->Other Languages->Visual C++项。因为我们要生成的是MFC程序,所以在“Visual C++”下选择“MFC”,对话框中间区域会出现三个选项:MFC ActiveX Control、MFC Application和MFC DLL。MFC ActiveX Control用来生成MFC ActiveX控件程序。MFC Application用来生成MFC应用程序。MFC DLL用来生成MFC动态链接库程序。当然我们要选择MFC Application。

在对话框下部有Name、Location和Solution name三个设置项。意义如下:Name--工程名,Location--解决方案路径,Solution name--解决方案名称。这里Name我们设为“HelloWorld”,Location设置为“桌面”的路径,Solution name默认和Name一样,当然可以修改为其他名字,这里我们不作修改,也使用“HelloWorld”。点“OK”按钮。

2.这时会弹出“MFC Application Wizard”对话框,上部写有“Welcome to the MFC Application

Wizard”,下面显示了当前工程的默认设置。第一条“Tabbed multiple document interface (MDI)”是说此工程是多文档应用程序。如果这时直接点下面的“Finish”按钮,可生成具有上面列出设置的多文档程序。但我们此例是要建立单文档应用程序,所以点“Next”按钮再继续设置吧。

3.接下来弹出的对话框上部写有“Application Type”,当然是让选择应用程序类型,我们看到有四种类型:Single document(单文档)、Multiple documents(多文档)、Dialog based(基于对话框)和Multiple top-level documents。我们选择Single document类型,以生成一个单文档应用程序框架。单文档应用程序运行时是一个单窗口界面。

此对话框的“Resource language”还提供语言的选择,这里默认选择英语。“Project style”可选择工程风格,我们选择默认的“Visual Studio”风格。“Use of MFC”有两个选项:Use MFC in a shared DLL(动态链接库方式使用MFC)和Use MFC in a static library(静态库方式使用MFC)。选择Use MFC in a shared DLL时MFC的类会以动态链接库的方式访问,所以我们的应用程序本身就会小些,但是发布应用程序时必须同时添加必要的动态链接库,以便在没有安装VS2010的机子上能够正常运行程序。选择Use MFC in a static library时MFC的类会编译到可执行文件中,所以应用程序的可执行文件要比上种方式大,但可以单独发布,不需另加包含MFC类的库。这里我们使用默认的Use MFC in a shared DLL。点“Next”按钮。

4.此时弹出上部写有“Compound Document Support”的对话框,可以通过它向应用程序加入OLE支持,指定OLE选项的复合文档类型。本例不需要OLE特性,使用默认值“None”。点“Next”按钮。

5.弹出的新对话框上部写有“Document Template Properties”。“File extension”可以设置程序能处理的文件的扩展名。对话框其他选项还可以更改程序窗口的标题。我们都使用默认设置,点“Next”按钮。

6.此时弹出的对话框主题是“Database Support”。用于设置数据库选项。此向导可以生成数据库应用程序需要的代码。它有四个选项:

None:忽略所有的数据库支持;

Header files only:只包含定义了数据库类的头文件,但不生成对应特定表的数据库类或视图类;

Database view without file support:创建对应指定表的一个数据库类和一个视图类,不附加标准文件支持;

Database view with file support:创建对应指定表的一个数据库类和一个视图类,并附加标准文件支持。

本例选择默认值“None”,不使用数据库特性。点“Next”按钮。

7.这时弹出的对话框是关于“User Interface Features”,即用户界面特性。我们可以设置有无最大化按钮、最小化按钮、系统菜单和初始状态栏等。还可以选择使用菜单栏和工具栏生成简单的应用程序还是使用ribbon。这里我们都选择默认设置。点“Next”进入下一步。

8.此时弹出“高级特性”对话框。可以设置的高级特性包括有无打印和打印预览等。在“Number of

files on recent file list”项可以设置在程序界面的文件菜单下面最近打开文件的个数。我们仍使用默认值。点“Next”按钮。

9.弹出“生成类”对话框。在对话框上部的“生成类”列表框内,列出了将要生成的4 个类:一个视图类(CHelloWorldView)、一个应用类(CHelloWorldApp)、一个文档类(CHelloWorldDoc)和一个主框架窗口类(CMainFrame)。在对话框下面的几个编辑框中,可以修改默认的类名、类的头文件名和源文件名。对于视图类,还可以修改其基类名称,默认的基类是CView,还有其他几个基类可以选择。这里我们还是使用默认设置。点“Finish”按钮。

应用程序向导最后为我们生成了应用程序框架,并在Solution Explorer中自动打开了解决方案(见上面第一张图)。

编译运行生成的程序

点菜单中的Build->Build HelloWorld编译程序,然后点Debug->Start Without Debugging(快捷键Ctrl+F5)运行程序,也可以直接点Debug->Start Without Debugging,这时会弹出对话框提示是否编译,选择“Yes”,VS2010将自动编译链接运行HelloWorld程序。结果页面如下所示:

终于看见界面了。鸡啄米在以后的教程中会继续讲解各种界面和控件的使用方法。欢迎到鸡啄米博客交流,您的关注是我前进的动力。

VS2010/MFC编程入门之三(VS2010应用程序工程中文件的组成结构)

分类标签: 编程入门 VC++ MFC VS2010

鸡啄米在上一讲中为大家演示了如何利用应用程序向导创建单文档应用程序框架。这一节将以上一讲中生成应用程序HelloWorld的文件结构为例,讲解VS2010应用程序工程中文件的组成结构。

用应用程序向导生成框架程序后,我们可以在之前设置的Location下看到以解决方案名命名的文件夹,此文件夹中包含了几个文件和一个以工程名命名的子文件夹,这个子文件夹中又包含了若干个文件和一个res文件夹,创建工程时的选项不同,工程文件夹下的文件可能也会有所不同。

如果已经以Debug方式编译链接过程序,则会在解决方案文件夹下和工程子文件夹下各有一个名为“Debug”的文件夹,而如果是Release方式编译则会有名为“Release”的文件夹。这两种编译方式将产生两种不同版本的可执行程序:Debug版本和Release版本。Debug版本的可执行文件中包含了用于调试的信息和代码,而Release版本则没有调试信息,不能进行调试,但可执行文件比较小。

鸡啄米将所有文件分为6个部分:解决方案相关文件、工程相关文件、应用程序头文件和源文件、资源文件、预编译头文件和编译链接生成文件。

1.解决方案相关文件

解决方案相关文件包括解决方案文件夹下的.sdf文件、.sln文件、.suo文件和ipch文件夹。

.sdf文件和ipch目录一般占用空间比较大,几十兆甚至上百兆,与智能提示、错误提示、代码恢复和团队本地仓库等相关。如果你觉得不需要则可以设置不生成它们,方法是点击菜单栏Tools->

Options,弹出Options对话框,选择左侧面板中Text Editor->C/C++->Advanced,右侧列表中第一项Disable Database由False改为True就可以了,最后关闭VS2010再删除.sdf文件和ipch目录以后就不会再产生了。但关闭此选项以后也会有很多不便,例如写程序时的智能提示没有了。

.sln文件和.suo文件为MFC自动生成的解决方案文件,它包含当前解决方案中的工程信息,存储解决方案的设置。

2.工程相关文件

工程相关文件包括工程文件夹下的.vcxproj文件和.s文件。

.vcxproj文件是MFC生成的工程文件,它包含当前工程的设置和工程所包含的文件等信息。.s文件存放工程的虚拟目录信息,也就是在解决方案浏览器中的目录结构信息。

3.应用程序头文件和源文件

应用程序向导会根据应用程序的类型(单文档、多文档或基于对话框的程序)自动生成一些头文件和源文件,这些文件是工程的主体部分,用于实现主框架、文档、视图等。鸡啄米下面分别简单介绍下各个文件:

HelloWorld.h:应用程序的主头文件。主要包含由CWinAppEx类派生的CHelloWorldApp类的声明,以及CHelloWorldApp类的全局对象theApp的声明。

:应用程序的主源文件。主要包含CHelloWorldApp类的实现,CHelloWorldApp类的全局对象theApp的定义等。

MainFrm.h和:通过这两个文件从CFrameWndEx类派生出CMainFrame类,用于创建主框架、菜单栏、工具栏和状态栏等。

HelloWorldDoc.h和:这两个文件从CDocument类派生出文档类CHelloWorldDoc,包含一些用来初始化文档、串行化(保存和装入)文档和调试的成员函数。

HelloWorldView.h和:它们从CView类派生出名为CHelloWorldView的视图类,用来显示和打印文档数据,包含了一些绘图和用于调试的成员函数。

ClassView.h和:由CDockablePane类派生出CClassView类,用于实现应用程序界面左侧面板上的Class View。

FileView.h和:由CDockablePane类派生出CFileView类,用于实现应用程序界面左侧面板上的File View。

OutputWnd.h和:由CDockablePane类派生出COutputWnd类,用于实现应用程序界面下侧面板Output。

PropertiesWnd.h和:由CDockablePane类派生出CPropertiesWnd类,用于实现应用程序界面右侧面板Properties。

ViewTree.h和:由CTreeCtrl类派生出CViewTree类,用于实现出现在ClassView和FileView等中的树视图。

4.资源文件

一般我们使用MFC生成窗口程序都会有对话框、图标、菜单等资源,应用程序向导会生成资源相关文件:res目录、文件和Resource.h文件。

res目录:工程文件夹下的res目录中含有应用程序默认图标、工具栏使用图标等图标文件。

:包含默认菜单定义、字符串表和加速键表,指定了默认的About对话框和应用程序默认图标文件等。

Resource.h:含有各种资源的ID定义。

5.预编译头文件

几乎所有的MFC程序的文件都要包含afxwin.h等文件,如果每次都编译一次则会大大减慢编译速度。所以把常用的MFC头文件都放到了stdafx.h文件中,然后由包含stdafx.h文件,编译器对只编译一次,并生成编译之后的预编译头,大大提高了编译效率。

6.编译链接生成文件

如果是Debug方式编译,则会在解决方案文件夹和工程文件夹下都生成Debug子文件夹,而如果是Release方式编译则生成Release子文件夹。

工程文件夹下的Debug或Release子文件夹中包含了编译链接时产生的中间文件,解决方案文件夹下的Debug或Release子文件夹中主要包含有应用程序的可执行文件。

关于应用程序工程文件的组成结构鸡啄米就先讲到这了。其中包含了很多专有名词,以后大家会慢慢熟悉的。欢迎来鸡啄米博客交流。谢谢。

VS2010/MFC编程入门之四(MFC应用程序框架分析)

分类标签: 编程入门 VC++ MFC VS2010

上一讲鸡啄米讲的是VS2010应用程序工程中文件的组成结构,可能大家对工程的运行原理还是很模糊,理不出头绪,毕竟跟C++编程入门系列中的例程差别太大。这一节鸡啄米就为大家分析下MFC应用程序框架的运行流程。

一.SDK应用程序与MFC应用程序运行过程的对比

程序运行都要有入口函数,在之前的C++教程中都是main函数,而Windows应用程序的入口函数是WinMain函数,MFC程序也是从WinMain函数开始的。下面鸡啄米就给出用Windows SDK写的“HelloWorld”程序,与应用程序框架进行对比,这样能更好的了解框架是怎样运行的。Windows

SDK开发程序就是不使用MFC类库,直接用Windows API函数进行软件开发。鸡啄米不是要讲解SDK开发,只是为了对比而简单介绍,至于SDK开发可以在大家学完MFC以后选择是否要研究,一般来说有简单了解就可以了。

SDK应用程序

首先,给出Windows SDK应用程序“HelloWorld”的源码:

C++代码

1.

2.

3.

4.

5.

6.

7.

8.

9.

#include

LRESULT CALLBACK myWndProc(HWND hWindow, UINT msg, WPARAM wParam, LPARAM lParam);

int WINAPI WinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, PSTR szCmdLine, int iCmdShow)

{

const static TCHAR appName[] = TEXT("Hello world");

WNDCLASSEX myWin;

= sizeof(myWin);

10. = CS_HREDRAW | CS_VREDRAW;

11. dProc = myWndProc;

12. xtra = 0;

13. xtra = 0;

14. nce = hInstance;

15. = 0;

16. m = 0;

17. r = 0;

18. kground = (HBRUSH)(COLOR_WINDOW + 1);

19. nuName = 0;

20. assName = appName;

21. //Register

22. if (!RegisterClassEx(&myWin)) return 0;

23. const HWND hWindow = CreateWindow(

24. appName,

25. appName,

26. WS_OVERLAPPEDWINDOW,

27. CW_USEDEFAULT,

28. CW_USEDEFAULT,

29. CW_USEDEFAULT,

30. CW_USEDEFAULT,

31. 0,

32. 0,

33. hInstance,

34. 0);

35. ShowWindow(hWindow,iCmdShow);

36. UpdateWindow(hWindow);

37. {

38. MSG msg;

39. while(GetMessage(&msg,0,0,0))

40. {

41. TranslateMessage(&msg);

42. DispatchMessage(&msg);

43. }

44. return (int);

45. }

46. }

47.

48. LRESULT CALLBACK myWndProc(HWND hWindow, UINT msg, WPARAM wParam, LPARAM lParam)

49. {

50. if (msg==WM_PAINT)

51. {

52. PAINTSTRUCT ps;

53. const HDC hDC = BeginPaint(hWindow,&ps);

54. RECT rect;

55. GetClientRect(hWindow,&rect);

56. DrawText(hDC,TEXT("HELLO WORLD"),-1,&rect, DT_SINGLELINE | DT_CENTER | DT_VCENTER);

57. EndPaint(hWindow,&ps);

58. return 0;

59. }

60. else if (msg==WM_DESTROY)

61. {

62. PostQuitMessage(0);

63. return 0;

64. }

65. return DefWindowProc(hWindow,msg,wParam,lParam);

66. }

上面的程序运行的流程是:进入WinMain函数->初始化WNDCLASSEX,调用RegisterClassEx函数注册窗口类->调用ShowWindow和UpdateWindow函数显示并更新窗口->进入消息循环。关于消息循环再简单说下,Windows应用程序是消息驱动的,系统或用户让应用程序进行某项操作或完成某个任务时会发送消息,进入程序的消息队列,然后消息循环会将消息队列中的消息取出,交予相应的窗口过程处理,此程序的窗口过程函数就是myWndProc函数,窗口过程函数处理完消息就完成了某项操作或任务。本例是要显示“HELLO WORLD”字符串,UpdateWindow函数会发送WM_PAINT消息,但是此消息不经过消息队列而是直接送到窗口过程处理,在窗口过程函数中最终绘制了“HELLO WORLD”字符串。

MFC应用程序

下面是MFC应用程序的运行流程,通过MFC库中代码进行分析:

首先在中定义全局对象theApp:CHelloWorldApp theApp;。调用CWinApp和CHelloWorldApp的构造函数后,进入WinMain函数(位于中)。

C++代码

1.

2.

3.

4.

5.

6.

extern "C" int WINAPI

_tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,

_In_ LPTSTR lpCmdLine, int nCmdShow)

#pragma warning(suppress: 4985)

{

// call shared/exported WinMain

7.

8.

return AfxWinMain(hInstance, hPrevInstance, lpCmdLine, nCmdShow);

}

在TCHAR.h中,有此定义:#define _tWinMain WinMain,所以这里的_tWinMain就是WinMain函数。它调用了AfxWinMain函数(位于中)。

C++代码

1.

2.

3.

4.

5.

6.

7.

8.

9.

int AFXAPI AfxWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,LPTSTR lpCmdLine, int nCmdShow)

{

.............略

// App global initializations (rare)

if (pApp != NULL && !pApp->InitApplication())

goto InitFailure;

if (!pThread->InitInstance())

{

10. .........略

11. }

12.

13. // Run函数位于中,由此函数进入消息循环

14. nReturnCode = pThread->Run();

15.

16. ..............略

17.

18. return nReturnCode;

19. }

上面InitInstance函数的代码如下:

C++代码

1.

2.

3.

4.

5.

6.

7.

8.

9.

BOOL CTestApp::InitInstance()

{

.............略

CSingleDocTemplate* pDocTemplate;

pDocTemplate = new CSingleDocTemplate(

IDR_MAINFRAME,

RUNTIME_CLASS(CTestDoc),

RUNTIME_CLASS(CMainFrame), // main SDI frame window

RUNTIME_CLASS(CTestView));

10. if (!pDocTemplate)

11. return FALSE;

12. AddDocTemplate(pDocTemplate);

13. // Parse command line for standard shell commands, DDE, file open

14.

15. CCommandLineInfo cmdInfo;

16. ParseCommandLine(cmdInfo);

17.

18. //ProcessShellCommand位于中,注册并创建窗口

19. if (!ProcessShellCommand(cmdInfo))

20. return FALSE;

21.

22. m_pMainWnd->ShowWindow(SW_SHOW);

23. m_pMainWnd->UpdateWindow();

24.

25. return TRUE;

26. }

InitInstance中的ProcessShellCommand函数又调用了CMainFrame的LoadFrame函数注册并创建了窗口,执行完ProcessShellCommand函数以后,调用了m_pMainWnd的ShowWindow和UpdateWindow函数显示并更新框架窗口。这些是不是与上面的SDK程序十分类似?

接下来该是消息循环了,上面的AfxWinMain函数中调用了pThread的Run函数(位于中),在Run中包含了消息循环。Run函数的代码如下:

C++代码

1.

2.

3.

4.

5.

6.

7.

8.

9.

int CWinThread::Run()

{

.............略

// phase2: pump messages while available

do

{

// pump message, but quit on WM_QUIT

if (!PumpMessage())

return ExitInstance();

10.

11. // reset "no idle" state after pumping "normal" message

12. if (IsIdleMessage(&m_msgCur))

13. {

14. bIdle = TRUE;

15.

16. lIdleCount = 0;

17.

18. }

19. } while (::PeekMessage(&m_msgCur, NULL, NULL, NULL, PM_NOREMOVE));

20. ..............略

21. }

22.

23. BOOL CWinThread::PumpMessage()

24. {

25. return AfxInternalPumpMessage();

26. }

27.

28. BOOL AFXAPI AfxInternalPumpMessage()

29. {

30. _AFX_THREAD_STATE *pState = AfxGetThreadState();

31.

32. if (!::GetMessage(&(pState->m_msgCur), NULL, NULL, NULL))

33. {

34. .............略

35. }

36. ...............略

37. if (pState->m_e != WM_KICKIDLE && !AfxPreTranslateMessage(&(pState->m_msgCur)))

38. {

39. ::TranslateMessage(&(pState->m_msgCur));

40. ::DispatchMessage(&(pState->m_msgCur));

41. }

42.

43. return TRUE;

44. }

我们看到PumpMessage中通过调用GetMessage、TranslateMessage、DispatchMessage等建立了消息循环并投递消息。

窗口过程函数AfxWinProc形式如下:

C++代码

1.

2.

3.

4.

5.

6.

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

{

……

CWnd*pWnd=CWnd::FromHandlePermanent(hWnd);

ReturnAfxCallWndProc(pWnd,hWnd,nMsg,wParam,lParam);

}

两者运行过程对比

到此,通过对比可以发现,MFC应用程序的运行流程与SDK程序是类似的,都是先进行一些初始化过程,再注册并创建窗口,然后显示、更新窗口,最后进入消息循环,消息都由窗口过程函数处理。现在大家是不是觉得有些头绪了?在运行流程上有基本的掌握即可。

二.MFC应用程序框架主要类之间的关系

在第二讲中,给大家演示了如何利用应用程序向导生成单文档应用程序框架,可以看到程序的基本框架和必要的代码都自动生成了,上一讲又讲解了文件组成结构,实际上在前面自动生成的框架中比较重要的类包括以下几个:CHelloWorldApp、CMainFrame、CHelloWorldDoc和CHelloWorldView,至于其他的类比如CClassView、CFileView等都是在框架窗口(CMainFrame)上创建的面板等,不是必要的。

鸡啄米就四个主要类的关系简单讲下,CHelloWorldApp类处理消息,将收到的消息分发给相应的对象。CMainFrame是视图CHelloWorldView的父窗口,视图CHelloWorldView就显示在CMainFrame的客户区中。视图类CHelloWorldView用来显示文档类CHelloWorldDoc中的数据,并根据对视图类的操作修改文档类的数据。一个视图类只能跟一个文档类相联系,而一个文档类可以跟多个视图类相联系。关于视图类和文档类的关系后面会详细讲解。

本节VC++/MFC编程入门教程内容比较多,主要是让大家对MFC应用程序的运行原理有大概的了解。对于以后的MFC开发有很多好处。如果有问题请在鸡啄米博客留言交流。谢谢。

首页 » 软件开发 » VS2010/MFC编程入门之五(MFC消息映射机制概述)

VS2010/MFC编程入门之五(MFC消息映射机制概述)

分类标签: 编程入门 VC++ MFC VS2010

上一讲鸡啄米为大家简单分析了MFC应用程序框架,这一讲是关于MFC消息映射机制的内容。

前面已经说过,Windows应用程序是消息驱动的。在MFC软件开发中,界面操作或者线程之间通信都会经常用到消息,通过对消息的处理实现相应的操作。比较典型的过程是,用户操作窗口,然后有消息产生,送给窗口的消息处理函数处理,对用户的操作做出响应。

什么是消息

窗口消息一般由三个部分组成:1.一个无符号整数,是消息值;(2)消息附带的WPARAM类型的参数;(3)消息附带的LPARAM类型的参数。其实我们一般所说的消息是狭义上的消息值,也就是一个无符号整数,经常被定义为宏。

什么是消息映射机制

MFC使用一种消息映射机制来处理消息,在应用程序框架中的表现就是一个消息与消息处理函数一一对应的消息映射表,以及消息处理函数的声明和实现等代码。当窗口接收到消息时,会到消息映射表中查找该消息对应的消息处理函数,然后由消息处理函数进行相应的处理。SDK编程时需要在窗口过程中一一判断消息值进行相应的处理,相比之下MFC的消息映射机制要方便好用的多。

Windows消息分类

先讲下Windows消息的分类。Windows消息分为系统消息和用户自定义消息。Windows系统消息有三种:

1.标准Windows消息。除WM_COMMAND外以WM_开头的消息是标准消息。例如,WM_CREATE、WM_CLOSE。

2.命令消息。消息名为WM_COMMAND,消息中附带了标识符ID来区分是来自哪个菜单、工具栏按钮或加速键的消息。

3.通知消息。通知消息一般由列表框等子窗口发送给父窗口,消息名也是WM_COMMAND,其中附带了控件通知码来区分控件。

CWnd的派生类都可以接收到标准Windows消息、通知消息和命令消息。命令消息还可以由文档类等接收。

用户自定义消息是实际上就是用户定义一个宏作为消息,此宏的值应该大于等于WM_USER,然后此宏就可以跟系统消息一样使用,窗口类中可以定义它的处理函数。

消息映射表

除了一些没有基类的类或CObject的直接派生类外,其他的类都可以自动生成消息映射表。下面的讲解都以前面例程HelloWorld的CMainFrame为例。消息映射表如下:

C++代码

1.

2.

3.

4.

5.

6.

7.

8.

BEGIN_MESSAGE_MAP(CMainFrame, CFrameWndEx)

ON_WM_CREATE()

ON_COMMAND(ID_VIEW_CUSTOMIZE, &CMainFrame::OnViewCustomize)

ON_REGISTERED_MESSAGE(AFX_WM_CREATETOOLBAR, &CMainFrame::OnToolbarCreateNew)

ON_COMMAND_RANGE(ID_VIEW_APPLOOK_WIN_2000, ID_VIEW_APPLOOK_WINDOWS_7, &CMainFrame::OnApplicationLook)

ON_UPDATE_COMMAND_UI_RANGE(ID_VIEW_APPLOOK_WIN_2000, ID_VIEW_APPLOOK_WINDOWS_7, &CMainFrame::OnUpdateApplicationLook)

ON_WM_SETTINGCHANGE()

END_MESSAGE_MAP()

在BEGIN_MESSAG_MAP和END_MESSAGE_MAP之间的内容成为消息映射入口项。消息映射除了在CMainFrame的实现文件中添加消息映射表外,在类的定义文件MainFrm.h中还会添加一个宏调用:

DECLARE_MESSAGE_MAP()

一般这个宏调用写在类定义的结尾处。

添加消息处理函数

如何添加消息处理函数呢?不管是自动还是手动添加都有三个步骤:

1.在类定义中加入消息处理函数的函数声明,注意要以afx_msg打头。例如MainFrm.h中WM_CREATE的消息处理函数的函数声明:afx_msg int OnCreate(LPCREATESTRUCT lpCreateStruct);。

2.在类的消息映射表中添加该消息的消息映射入口项。例如WM_CREATE的消息映射入口项:ON_WM_CREATE()。

3.在类实现中添加消息处理函数的函数实现。例如,中WM_CREATE的消息处理函数的实现:

int CMainFrame::OnCreate(LPCREATESTRUCT lpCreateStruct)

{

......

}

通过以上三个步骤以后,WM_CREATE等消息就可以在窗口类中被消息处理函数处理了。

各种Windows消息的消息处理函数

标准Windows消息的消息处理函数都与WM_CREATE消息类似。

命令消息的消息映射入口项形式如:ON_COMMAND(ID_VIEW_CUSTOMIZE, &CMainFrame::OnViewCustomize),消息为ID_VIEW_CUSTOMIZE,消息处理函数为OnViewCustomize。

如果想要使用某个处理函数批量处理某些命令消息,则可以像CMainFrame消息映射表中的ON_COMMAND_RANGE(ID_VIEW_APPLOOK_WIN_2000, ID_VIEW_APPLOOK_WINDOWS_7,

&CMainFrame::OnApplicationLook)一样添加消息映射入口项,这样值在ID_VIEW_APPLOOK_WIN_2000到ID_VIEW_APPLOOK_WINDOWS_7之间的菜单项等的命令消息都由CMainFrame的OnApplicationLook函数处理。函数原型为afx_msg void OnApplicationLook(UINT id);,参数id为用户操作的菜单项等的ID。

在操作列表框等控件时往往会给父窗口发送WM_NOTIFY通知消息。WM_NOTIFY消息的wParam参数为发送通知消息的控件的ID,lParam参数指向一个结构体,可能是NMHDR结构体,也可能是第一个元素为NMHDR结构体变量的其他结构体。NMHDR结构体的定义如下(仅作了解):

Typedef sturct tagNMHDR{

HWND hwndFrom;

UINT idFrom;

UINT code;

} NMHDR;

hwndFrom为发送通知消息控件的句柄,idFrom为控件ID,code为要处理的通知消息的通知码,例如NM_CLICK。

通知消息的消息映射入口项形式如:

ON_NOTIFY(wNotifyCode,id,memberFxn)

wNotifyCode为要处理的通知消息通知码,例如:NM_CLICK。id为控件标识ID。MemberFxn为此消息的处理函数。

通知消息的处理函数的原型为:

afx_msg void memberFxn( NMHDR * pNotifyStruct, LRESULT * result);

如果需要使用用户自定义消息,首先要定义消息宏,如:#define WM_UPDATE_WND (WM_USER+1),再到消息映射表中添加消息映射入口项:ON_MESSAGE(WM_UPDATE_WND, &CMainFrame::OnUpdateWnd),然后在MainFrm.h中添加消息处理函数的函数声明:afx_msg LRESULT

OnUpdateWnd(WPARAM wParam, LPARAM lParam);,最后在中实现此函数。

鸡啄米本节对MFC消息映射机制只是做了比较简单的讲解,让大家对它有一定的认识,编程入门者不必强求完全掌握。在以后的教程中会经常涉及到消息的使用,大家会逐渐熟悉MFC的消息映射机制。