C++MFC编程笔记day01 MFC介绍、创建MFC程序和重写消息处理
一、MFC概念和作用
1、全称Microsoft Foundation Class Library,我们称为微软基础类库,封闭了绝大部分的win32 Api函数,C++语法中的数据结构,程序的运行流程
MFC就是一个库(动态库,静态库)
MFC还是一个程序框架
2、为什么使用MFC
基于框架编程。提高工作效率,降低开发周期,节约开发成本。
二、几个重要的头文件
afx.h -绝大部分类的声明头文件
afxwin.h -包括了afx.h和windows.h
afxext.h -提供了扩展窗体类的支持(工具栏,状态栏等)
三、使用MFC
1、使用MFC库做自己的控制台程序
入口函数 _tmain()不同于以往的main()
CWinApp theApp;-全局程序对象。
注意:
::函数名()-为win32的API函数。
以Afx开头的函数,为MFC库里的全局函数。
2、使用MFC库做自己的库程序(MFC AppWizard dll)
规则动态库:能够被不论什么程序调用
扩展动态库:仅仅能被支持MFC的程序调用
3、MFC窗体程序
单文档视图构架程序。
CFrameWnd-框架窗体类;
CWinApp-应用程序类,负责管理整个程序的运行流程;
CDocument-文档类,负责管理数据(数据的提取、转换、存储等操作);
CView-视图窗体类,负责显示数据
多文档视图构架程序。
CMDIChildWnd-子框架窗体类,负责管理子框架窗体。
CMDIFrameWnd-主框架窗体类
CWinApp-应用程序类,负责管理整个程序的运行流程;
CDocument-文档类,负责管理数据(数据的提取、转换、存储等操作);
CView-视图窗体类,负责显示数据
对话框程序
CWinApp-应用程序类,负责管理整个程序的运行流程;
CDialog-对话框类
四、MFC类的介绍
CObject -mfc库绝大部分类的基类,封装了MFC库中最主要的一些机制,执行时类信息机制/动态创建机制/序列化机制.
CCmdTarget-消息映射机制基类
CWinThread/CWinApp-应用程序类
CDocTemplate及其子类 -文档模版类
CDocument及其子类-文档类,封装了对各种数据的操作。
Exceptions - 异常类,封装了MFC中的各种异常情况。
CFile-文操作类
CWnd -全部窗体类基类
FrameWindows 框架窗体类,封装了对各种框架窗体的操作
ControlBars-各种工具栏类,封闭了对各种工具栏的操作
Dialog Boxes 对话框类。
Views 视图窗体类
Controls 控件类
CDC/CGdiObject 封装了画图操作
CArray/AList/CMap 对C++中的数据的封装,以C开头。
机制1:MFC入口函数机制
如今学习时。MFC都是建立Win32 Application程序。然后通过下列步骤改为MFC程序,仅仅有一步步了解怎么去手动创建MFC,以后通过MFC向导自己主动生成的大量MFC代码我们才干看得懂。
1、删除主程序cpp中的入口函数
2、将stdafx.h中的windows.h改为afxwin.h
3、project-设置 中选择使用MFC库
4、自己定义框架类和程序类,继承自MFC类,并重写CWinApp的InitInstance方法:
class CMyFrameWnd:public CFrameWnd
{
};
class CMyWinApp:public CWinApp
{
public:
CMyWinApp();
virtual BOOL InitInstance();
};
CMyWinApp::CMyWinApp()
{
}
BOOL CMyWinApp::InitInstance()
{
CMyFrameWnd *pFrame=new CMyFrameWnd;
pFrame->Create(NULL,"MFC base");
m_pMainWnd=pFrame;
pFrame->ShowWindow(SW_SHOW);
pFrame->UpdateWindow();
return true;
}
5、创建一个程序类对象(触发点):
CMyWinApp theApp;
AFX_MODULE_STATE* pModuleState = AfxGetModuleState();//获取当前程序模块状态信息
AFX_MODULE_THREAD_STATE* pThreadState = pModuleState->m_thread;//获取当前线程状态信息
pThreadState->m_pCurrentWinThread = this;//theapp全局对象信息赋值
pModuleState->m_pCurrentWinApp = this;
ASSERT(bool表达式);//假设表达式不成立,则弹出错误,不往下运行
全局函数:AfxGetApp(),AfxGetThread() 获取theApp地址
AfxMessageBox();//弹出消息
vc6中。查看-调试窗体-call stack 能够看到被谁调用
可重写(前加virtual):
BOOL CMyWinApp::InitApplication();
BOOL CMyWinApp::InitInstance()//初始化
int CMyWinApp::Run()//执行
BOOL CMyWinApp::OnIdle(LONG lcount);//空暇时处理
int CMyWinApp::ExitInstance();
演示样例代码:
// MFCbase.cpp : Defines the entry point for the application.
// #include "stdafx.h"
class CMyFrameWnd:public CFrameWnd
{ };
class CMyWinApp:public CWinApp
{
public: CMyWinApp();
virtual BOOL InitInstance();
virtual int Run();
};
CMyWinApp::CMyWinApp()
{ }
BOOL CMyWinApp::InitInstance()
{
CMyFrameWnd *pFrame=new CMyFrameWnd;
pFrame->Create(NULL,"MFC base");
m_pMainWnd=pFrame;
pFrame->ShowWindow(SW_SHOW);
pFrame->UpdateWindow();
return true;
}
int CMyWinApp::Run()
{
AfxMessageBox("run");
return CWinApp::Run();
}
CMyWinApp theApp;//创建一个程序对象
机制2:MFC窗体创建机制:
AfxGetInstanceHandle();//全局句柄
1、载入菜单
2、调用CreateEx 设计、注冊窗体类
PreCreateWindow():
WNDCLASS wndcls;wndcls.lpfWinProc=DefWindowProc;
AfxRegisterWithIcon(),AfxRegisterClass()
::RegisterClass注冊一个局部窗体类。类名“AfxFrameOrView42sd”,默认处理函数。
AfxHookWindowCreate(pFrame) 埋下勾子
CHandleMap* pMap = afxMapHWND(TRUE);
AFX_MODULE_THREAD_STATE* pState = AfxGetModuleThreadState();//映射对象保存到全局变量
pState->m_pmapHWND = new CHandleMap(...);
pMap->SetPermanent(m_hWnd = hWndNew, this);
//pFrame->m_hWnd 为窗体句柄。
CWnd* pWnd=CWnd::FromHandlePermanent(hWnd);
//pState->m_pmapHWND->LookupPermanent(HANDLE h)成员,可通过窗体句柄获取pFrame
::CreateWindowEx函数,创建窗体
3、钩子处理函数
将框架对象地址和窗体句柄建立一一相应关系。
利用::SetWindowLong,将窗体处理函数更改为AfxWndProc.
----------MSDN Library Visual Studio 6.0
//埋下勾子
HHOOK SetWindowsHookEx(
int idHook, // 勾子类型
HOOKPROC lpfn, // 处理函数(不同类型不同)
HINSTANCE hMod, // 进程句柄,NULL时不限制范围
DWORD dwThreadId // 线程ID
);
//勾子类型为WH_CBT时的处理函数
LRESULT CALLBACK CBTProc(
int nCode, // hook code
WPARAM wParam, // 创建好的窗体句柄
LPARAM lParam // depends on hook code
);
//勾子处理函数改动默认处理函数
WNDPROC afxWndProc = AfxGetAfxWndProc();
oldWndProc = (WNDPROC)SetWindowLong(hWnd, GWL_WNDPROC,(DWORD)afxWndProc);
//改动窗体处理函数
LONG SetWindowLong(
HWND hWnd, // 窗体句柄
int nIndex, // 传入:GWL_WNDPROC
LONG dwNewLong // 窗体处理函数地址
);
//全局变量信息
_AFX_THREAD_STATE* pThreadState=_afxThreadState.GetDate();
::GetCurrentThreadId();//获取当前线程ID
pThreadState->m_pWndInit = pWnd;//赋值到全局变量
::CreateWindowEx() //此函数处理完后直接进入勾子处理函数
重写消息处理函数
LRESULT CMyFrameWnd::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
switch(message)
{
case WM_CREATE:
AfxMessageBox("WM_CREATE");
break;
}
return CFrameWnd::WindowProc(message,wParam,lParam);
}
演示样例:
// MFCcreate2.cpp : Defines the entry point for the application.
#include "stdafx.h" class CMyFrameWnd:public CFrameWnd
{
public:
virtual LRESULT WindowProc(UINT message, WPARAM wParam, LPARAM lParam);
};
LRESULT CMyFrameWnd::WindowProc(UINT message, WPARAM wParam, LPARAM lParam)
{
switch(message)
{
case WM_CREATE:
AfxMessageBox("WM_CREATE");
break;
case WM_PAINT:
{
PAINTSTRUCT ps={0};
HDC hdc=::BeginPaint(this->m_hWnd,&ps);
::TextOut(hdc,100,100,"hello",5);
::EndPaint(this->m_hWnd,&ps);
}
break;
case WM_MOUSEMOVE:
{
//g_xpos=LOWORD(lParam);
//g_ypos=HIWORD(lParam);
::InvalidateRect(this->m_hWnd,NULL,true);
}
break;
}
return CFrameWnd::WindowProc(message,wParam,lParam);
}
class CMyWinApp:public CWinApp
{
public:
virtual BOOL InitInstance();
};
BOOL CMyWinApp::InitInstance()
{
CMyFrameWnd *pFrame=new CMyFrameWnd;
pFrame->Create(NULL,"MFCcreate");
m_pMainWnd=pFrame;
pFrame->ShowWindow(SW_SHOW);
pFrame->UpdateWindow();
return TRUE;
//return CWinApp::InitInstance();
}
CMyWinApp theApp;
C++MFC编程笔记day01 MFC介绍、创建MFC程序和重写消息处理的更多相关文章
- VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)
VS2010/MFC编程入门之二(利用MFC向导生成单文档应用程序框架)-软件开发-鸡啄米 http://www.jizhuomi.com/software/141.html 上一讲中讲了VS20 ...
- C++MFC编程笔记day05 文档类-单文档和多文档应用程序
文档类 1 相关类 CDocument类-父类是CCmdTarget类,所以,文档类也能够处理菜单等 命令消息. 作用保存和管理数据. 注意事项:怎样解决断言错 ...
- C++MFC编程笔记day06 MFC向导、MFC画图类使用
MFC画图 MFC画图类包含画图设备类和画图对象类 1 画图设备类 CDC类-父类是CObject,封装的是一般的画图设备,比如:显示器, 打印机等. ...
- VS2010/MFC编程入门之四十四(MFC常用类:定时器Timer)
前面一节鸡啄米讲了CTime类和CTimeSpan类的使用,本节继续讲与时间有关的定时器.定时器并不是一个类,主要考虑到,提起时间的话就不能不说定时器,所以就把它放到CTime和CTimeSpan之后 ...
- VS2010/MFC编程入门之四十六(MFC常用类:MFC异常处理)
上一节中鸡啄米讲了CFile文件操作类,本节主要来说说MFC异常处理. 在鸡啄米C++编程入门系列的最后一节鸡啄米:C++编程入门系列之五十(异常处理)中,鸡啄米讲了C++标准异常的处理机制,如果你还 ...
- MFC学习笔记(一): 不用MFC向导如何新建一个MFC程序
使用Visual Studio新建一个空项目,项目命名为HelloMFC,完成后,打开项目属性页面,将配置属性选项卡中的常规项打开,将其中的MFC的使用属性栏改为:在静态库中使用MFC或者在共享DLL ...
- C++MFC编程笔记day03 MFC工具栏、状态栏、视图窗体
MFC工具栏 相关类: CToolBarCtrl - 父类是 CWnd 封装了工具栏控件相关操作 CToolBar - 父类是CControlBar 封装了工具栏和框架窗体之间的关系 工具栏使用: ...
- VS2010/MFC编程入门之四十五(MFC常用类:CFile文件操作类)
上一节中鸡啄米讲了定时器Timer的用法,本节介绍下文件操作类CFile类的使用. CFile类概述 如果你学过C语言,应该知道文件操作使用的是文件指针,通过文件指针实现对它指向的文件的各种操作.这些 ...
- MFC编程入门之九(对话框:为控件添加消息处理函数)
这一节讲的主要内容是如何为控件添加消息处理函数. MFC为对话框和控件定义了诸多消息,我们对他们操作时会触发消息,这些消息最终由消息处理函数处理,比如我们点击按钮时就会产生BN_CLICKED消息,修 ...
随机推荐
- oracle 表之间的连接
排序 - - 合并连接(Sort Merge Join, SMJ): a) 对于非等值连接,这种连接方式的效率是比较高的. b) 如果在关联的列上都有索引,效果更好. c) 对于将2个较大的row s ...
- CAD嵌套打印(com接口版)
当用户需要打印两个CAD控件的图纸时,可以采用嵌套打印实现.实现嵌套打印功能,首先将两个CAD控件放入网页中,C#代码如下: private void BatchPrintDialog() { MxD ...
- LPCTSTR 字符串获取其长度
LPCTSTR lpStr = "123456789";int i=CString(lpStr).GetLength();
- JavaSE-06 二维数组
学习要点 二维数组的定义 二维数组内存数据结构 不规则二维数组 二维数组的定义 语法格式 格式一 数据类型[][] 数组名 = new 数据类型[m][n]; m:表示这个二维数组有多少个一维数组. ...
- Android全局异常捕获
PS:本文摘抄自<Android高级进阶>,仅供学习使用 Java API提供了一个全局异常捕获处理器,Android引用在Java层捕获Crash依赖的就是Thread.Uncaught ...
- joda-time时间操作组件
今天看到了学习到了一个不错的操作时间的jar包,很方便的,以后操作时间运算就可以直接使用jar包中的方法了,再也不用自己写操作时间的方法了.懒的不行不行的 <!-- 时间操作组件 --> ...
- Java代码的编译和执行
Java代码编译和执行的整个过程包含了以下三个重要的机制: (1)Java源码编译机制 (2)类加载机制 (3)类执行机制 1.Java代码编译是由Java源码编译器来完成,流程图: Java 源码编 ...
- 【牛客小白月赛6】 C 桃花 - 树上最长路
题目地址:https://www.nowcoder.com/acm/contest/136/C dfs找出最长路和次长路,将两个结果相加再加上起点即可: #include<iostream> ...
- Python re模块 subprocess模块
re模块 内部实现不是Python 而是调用了c的库 re是什么 正则 表达 式子 就是一些带有特殊含义的符号或者符号的组合作用: 对字符串进行过滤 在一对字符串中找到所关心的内容 你就需要告诉计算机 ...
- [Python3网络爬虫开发实战] 1.2.4-GeckoDriver的安装
上一节中,我们了解了ChromeDriver的配置方法,配置完成之后便可以用Selenium驱动Chrome浏览器来做相应网页的抓取. 那么对于Firefox来说,也可以使用同样的方式完成Seleni ...