1,寻找WinMain人口:

在安装目录下找到MFC文件夹下的SRC文件夹,SRC下是MFC源代码。

路径:MFC|SRC|APPMODUL.CPP:

_tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,

LPTSTR lpCmdLine, int nCmdShow)

{

// call shared/exported WinMain

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

}

注意:(#define _tWinMain WinMain)

2,对于全局对象或全局变量来说,在程序运行即WINMAIN函数加载的时候,已经为全局对象或全局变量分配了内存和赋初值。

所以:CTEApp theApp;->CTEApp ::CTEApp(){}->_tWinMain(){}

说明:每一个MFC程序,有且只有一个从WinApp类派生的类(应用程序类),也只有一个从应用程序类所事例化的对象,表示应用程序本身

。在WIN32程序当中,表示应用程序是通过WINMAIN入口函数来表示的(通过一个应用程序的一个事例号这一个标识来表示的)。在基于MFC应用

程序中,是通过产生一个应用程序对象,用它来唯一的表示了应用程序。

3,通过构造应用程序对象过程中调用基类CWinApp的构造函数,在CWinApp的构造函数中对程序包括运行时一些初始化工作完成了。

CWinApp构造函数:MFC|SRC|APPCORE.CPP

CWinApp::CWinApp(LPCTSTR lpszAppName){...}//带参数,而CTEApp构造函数没有显式向父类传参,难道CWinApp()有默认参数?见下:

(在CWinApp类定义中,CWinApp(LPCTSTR lpszAppName = NULL);)

注意:CWinApp()函数中:

pThreadState->m_pCurrentWinThread = this;

pModuleState->m_pCurrentWinApp = this

(this指向的是派生类CTEApp对象,即theApp)

调试:CWinApp::CWinApp();->CTEApp theApp;(->CTEApp ::CTEApp())->CWinApp::CWinApp()->CTEApp ::CTEApp()->_tWinMain(){}

4,_tWinMain函数中通过调用AfxWinMain()函数来完成它要完成的功能。(Afx*前缀代表这是应用程序框架函数,是一些全局函数,应用程序

框架是一套辅助生成应用程序的框架模型,把一些类做一些有机的集成,我们可根据这些类函数来设计自己的应用程序)。

AfxWinMain()函数路径:MFC|SRC|WINMAIN.CPP:

在AfxWinMain()函数中:

CWinApp* pApp = AfxGetApp();

说明:pApp存储的是指向WinApp派生类对象(theApp)的指针。

//_AFXWIN_INLINE CWinApp* AFXAPI AfxGetApp()

// { return afxCurrentWinApp; }

调用pThread->InitInstance()

说明:pThread也指向theApp,由于基类中virtual BOOL InitApplication()定义为虚函数,所以调用pThread->InitInstance()时候,调用的是

派生类CTEApp的InitInstance()函数。

nReturnCode = pThread->Run();

说明:pThread->Run()完成了消息循环。

5,注册窗口类:AfxEndDeferRegisterClass();

AfxEndDeferRegisterClass()函数所在文件:MFC|SRC|APPCORE.CPP

BOOL AFXAPI AfxEndDeferRegisterClass(LONG fToRegister){...}

说明:设计窗口类:在MFC中事先设计好了几种缺省的窗口类,根据不同的应用程序的选择,调用AfxEndDeferRegisterClass()函数注册所选择的窗口类。

调试:

CWinApp::CWinApp();->CTEApp theApp;(->CTEApp ::CTEApp())->CWinApp::CWinApp()->CTEApp ::CTEApp()->_tWinMain(){}//进入程序

->AfxWinMain();->pApp->InitApplication();->pThread->InitInstance()//父类InitInstance虚函数;

->CTEApp::InitInstance()//子类实现函数;

->AfxEndDeferRegisterClass(LONG fToRegister)//注册所选择的窗口类(出于文档管理,注册提前,正常的应在PreCreateWindow中进行注册

)//之后进入创建窗口阶段(以下再不做调试)

6,PreCreateWindow()://主要是注册窗口类

BOOL CMainFrame::PreCreateWindow(CREATESTRUCT& cs)

{

if( !CFrameWnd::PreCreateWindow(cs) )

return FALSE;

return TRUE;

}

说明:

CFrameWnd::PreCreateWindow()函数所在文件:MFC|SRC|WINFRM.CPP

BOOL CFrameWnd::PreCreateWindow(CREATESTRUCT& cs)

{

if (cs.lpszClass == NULL)

{

VERIFY(AfxDeferRegisterClass(AFX_WNDFRAMEORVIEW_REG));

//判断AFX_WNDFRAMEORVIEW_REG型号窗口类是否注册,如果没有注册则注册

cs.lpszClass = _afxWndFrameOrView; // COLOR_WINDOW background

//把注册后的窗口类名赋给cs.lpszClass

}

if ((cs.style & FWS_ADDTOTITLE) && afxData.bWin4)

cs.style |= FWS_PREFIXTITLE;

if (afxData.bWin4)

cs.dwExStyle |= WS_EX_CLIENTEDGE;

return TRUE;

}

其中:

virtual BOOL PreCreateWindow(CREATESTRUCT& cs);//PreCreateWindow()是个虚函数,如果子类有则调用子类的。

#define VERIFY(f) ASSERT(f)

#define AfxDeferRegisterClass(fClass) AfxEndDeferRegisterClass(fClass)

define AFX_WNDFRAMEORVIEW_REG 0x00008

const TCHAR _afxWndFrameOrView[] = AFX_WNDFRAMEORVIEW;//WINCORE.CPP文件中,定义为全局数组。

//#define AFX_WNDFRAMEORVIEW AFX_WNDCLASS("FrameOrView")

7,创建窗口:

Create()函数路径:MFC|SRC|WINFRM.CPP:

CFrameWnd::Create(...){

...

CreateEx(...);//从父类继承来的,调用CWnd::CreateEx().

...

}

CWnd::CreateEx()函数路径:MFC|SRC|WINCORE.CPP

BOOL CWnd::CreateEx(...){

...

if (!PreCreateWindow(cs))//虚函数,如果子类有调用子类的。

{

PostNcDestroy();

return FALSE;

}

...

HWND hWnd = ::CreateWindowEx(cs.dwExStyle, cs.lpszClass,

cs.lpszName, cs.style, cs.x, cs.y, cs.cx, cs.cy,

cs.hwndParent, cs.hMenu, cs.hInstance, cs.lpCreateParams);

...

}

说明:CreateWindowEx()函数与CREATESTRUCT结构体参数的对应关系,使我们在创建窗口之前通过可PreCreateWindow(cs)修改cs结构体成员来

修改所要的窗口外观。PreCreateWindow(cs))//是虚函数,如果子类有调用子类的。

HWND CreateWindowEx(

DWORD dwExStyle,

LPCTSTR lpClassName,

LPCTSTR lpWindowName,

DWORD dwStyle,

int x,

int y,

int nWidth,

int nHeight,

HWND hWndParent,

HMENU hMenu,

HINSTANCE hInstance,

LPVOID lpParam

);

typedef struct tagCREATESTRUCT { // cs

LPVOID lpCreateParams;

HINSTANCE hInstance;

HMENU hMenu;

HWND hwndParent;

int cy;

int cx;

int y;

int x;

LONG style;

LPCTSTR lpszName;

LPCTSTR lpszClass;

DWORD dwExStyle;

} CREATESTRUCT;

8,显示和更新窗口:

CTEApp类,TEApp.cpp中

m_pMainWnd->ShowWindow(SW_SHOW);//显示窗口,m_pMainWnd指向框架窗口

m_pMainWnd->UpdateWindow();//更新窗口

说明:

class CTEApp : public CWinApp{...}

class CWinApp : public CWinThread{...}

class CWinThread : public CCmdTarget

{ ...

public:

CWnd* m_pMainWnd;

...}

9,消息循环:

int AFXAPI AfxWinMain()

{ ...

// Perform specific initializations

if (!pThread->InitInstance()){...}

//完成窗口初始化工作,完成窗口的注册,完成窗口的创建,显示和更新。

nReturnCode = pThread->Run();

//继承基类Run()方法,调用CWinThread::Run()来完成消息循环...

}

CWinThread::Run()方法路径:MFC|SRC|THRDCORE.CPP

int CWinThread::Run()

{ ...

// phase2: pump messages while available

do//消息循环

{

// pump message, but quit on WM_QUIT

if (!PumpMessage())//取消息并处理

return ExitInstance(); ...

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

...

}

说明:

BOOL PeekMessage(,,,,)函数说明

The PeekMessage function checks a thread message queue for a message and places the message (if any) in the specified

structure.

If a message is available, the return value is nonzero.

If no messages are available, the return value is zero.

/

BOOL CWinThread::PumpMessage()

{ ...

if (!::GetMessage(&m_msgCur, NULL, NULL, NULL))//取消息

{...} ...

// process this message

if (m_msgCur.message != WM_KICKIDLE && !PreTranslateMessage(&m_msgCur))

{

::TranslateMessage(&m_msgCur);//进行消息(如键盘消息)转换

::DispatchMessage(&m_msgCur);//分派消息到窗口的回调函数处理(实际上分派的消息经过消息映射,交由消息响应函数进行处理。)

}

return TRUE;

}

10,文档与视结构:

可以认为View类窗口是CMainFram类窗口的子窗口。

DOCument类是文档类。

DOC-VIEW结构将数据本身与它的显示分离开。

文档类:数据的存储,加载

视类:数据的显示,修改

11,文档类,视类,框架类的有机结合:

在CTEApp类CTEApp::InitInstance()函数中通过文档模板将文档类,视类,框架类的有机组织一起。...

CSingleDocTemplate* pDocTemplate;

pDocTemplate = new CSingleDocTemplate(

IDR_MAINFRAME,

RUNTIME_CLASS(CTEDoc),

RUNTIME_CLASS(CMainFrame), // main SDI frame window

RUNTIME_CLASS(CTEView));

AddDocTemplate(pDocTemplate);//增加到模板...

----------------------------------------------------------------------------->

//AfxWinMain()函数在WINMAIN.CPP文件中,它主要调用以下函数

AfxWinInit();

pApp->InitApplication(); //内部初始化管理

pThread->InitInstance(); //调用子类中的InitInstance()

CTestApp::InitInstance();

┣━ProcessShellCommand(cmdInfo); //对命令行进行解释

┃ CTestDoc::CTestDoc(); //构造文档类对象

┃ CMainFrame::CMainFrame(); //构造框架窗口对象

┃ CFrameWnd::LoadFrame(); //WINFRM.CPP

┃ ┣━AfxEndDeferRegisterClass(); //WINCORE.CPP,注册窗口类

┃ ┃ AfxRegisterClass(); //WINCORE.CPP

┃ ┣━CMainFrame::PreCreateWindow();

┃ ┃ CFrameWnd::PreCreateWindow();

┃ ┃ AfxEndDeferRegisterClass();

┃ ┣━AfxRegisterClass();

┃ ┗━CFrameWnd::Create(); //创建CMainFrame窗口

┃ CWnd::CreateEx();

┃ CMainFrame::PreCreateWindow();

┃ CFrameWnd::PreCreateWindow();

┃ CTestView::CTestView(); //构造CTestView对象

┃ CWnd::CreateEx(); //创建CTestView窗口

┃ AfxEndDeferRegisterClass();

┃ AfxEndDeferRegisterClass();

┃ CWnd::CreateEx(); //创建CToolBar工具栏

┃ AfxEndDeferRegisterClass();

┃ CWnd::CreateEx(); //创建CStatusBar状态栏

┃ AfxEndDeferRegisterClass();

┃ AfxRegisterClass();

┃ CWnd::CreateEx(); //创建CDockBar

┃ AfxEndDeferRegisterClass();

┃ CWnd::CreateEx(); //创建CDockBar

┃ AfxEndDeferRegisterClass();

┃ CWnd::CreateEx(); //创建CDockBar

┃ AfxEndDeferRegisterClass();

┃ CWnd::CreateEx(); //创建CDockBar

┣━m_pMainWnd->ShowWindow(SW_SHOW); //显示窗口

┗━m_pMainWnd->UpdateWindow(); //更新窗口

nReturnCode = pThread->Run(); //进入消息循环

mfc编程 孙鑫_孙鑫VC++视频教程笔记-(3)MFC程序框架的剖析 附1-SDI程序流程图相关推荐

  1. mfc编程 孙鑫_孙鑫c++视频教程百度网盘 | 软件库

    第1章  Windows程序内部运行机制 1.1  API与SDK 1.2  窗口与句柄 1.3  消息与消息队列 1.4  WinMain函数 1.4.1  WinMain函数的定义 1.4.2   ...

  2. MFC程序框架的剖析

    和Win32平台创建Windows程序作对比: MFC有个theApp全局变量来代表程序的本身. 1.WinMain 寻找WinMain入口: 在安装目录下找到MFC文件夹下的SRC文件夹,SRC下是 ...

  3. mfc编程 孙鑫_孙鑫MFC学习笔记6:菜单编程

    1.对菜单响应的顺序: 视类,文档类,框架类,应用程序类 2.消息的分类 3.CWnd继承自CCmdTarget类, 所以从CWnd派生出的类也可以接收WM_COMMAND消息 4.命令的消息路由 5 ...

  4. mfc编程 孙鑫_孙鑫MFC学习笔记7:对话框编程(上)

    1.DoModal创建模态对话框 2.Create创建非模态对话框(需要用ShowWindow显示出来) 模态:对话框显示,程序会暂停,直到对话框关闭 非模态:对话框显示,程序继续执行 3.对于模态对 ...

  5. VC学习笔记---ATL MFC CLR三个库的区别

    MFC.ATL和CLR是VC2005内置的三大库,涵盖了Windows的各种开发方法和开发应用.当然关于C++开发的库不止这三个,不过这三个是微软推荐. 从编程所处层次而言,WIN32为最底层,其次是 ...

  6. c语言编程软件平板_想在ipad上进行C语言程序编写,请问有没有编译的APP

    app store 里搜索 C++ Programming Languageapp store 里搜索 C++ Programming Language 本回答被提问者采纳 app store 里搜索 ...

  7. hdc mfc 画扇形图_科学网—画扇形图(idl程序) - 张国印的博文

    IDL画扇形图还是有些麻烦的,今天中午没午休,以红移和RA为例写了程序,希望以后能用上 pro sector set_plot,'ps' device,file='F:Aprilmap.ps' REA ...

  8. java面向对象编程集合边框_第六章使用java实现面向对象-集合框架

    一:接口:即表示集合的抽象数据类型. 实现:即集合框架中接口的实现. 算法:在一个实现了某个集合框架中的接口的对象身上完成某种有用的计算的方法,例如查找. 排序等. Collection 接口存储一组 ...

  9. VS2019/MFC编程入门之对话框:向导对话框的创建及显示

    上一讲讲了属性页对话框和相关的两个类CPropertyPage类和CPropertySheet类,对使用属性页对话框做准备.本节将为大家演示如何创建向导对话框. 仍然以前面的"加法计算器&q ...

最新文章

  1. 忘了root口令解决方法
  2. MATLAB读取excel文件中的数据
  3. Hibernate--使用xml配置映射关系
  4. Linux进程 excel族函数的用法
  5. Android 如何让EditText不自动获取焦点
  6. Programe_Of_Beauty:2.14 求数组的子数组之和的最大值
  7. PAID Network宣布已获币安DeFi加速器基金投资
  8. Servlet获取全路径
  9. pthread_create()在C和C++使用区别
  10. 微信撤回软件安卓版_微信强制撤回软件下载-微信强制撤回消息工具(不限时间) v1.0安卓版_5577安卓网...
  11. 安卓吃鸡玩家专属:教你电脑玩刺激战场匹配手机最简单的方式
  12. 【已解决】【Selenium】请教大神,知乎的注册页面如何切换到登录页面?
  13. 邮件内容安全防护之反垃圾邮件开源软件ASSP
  14. 使用D3渲染中国地图
  15. 2017 CCCC预赛总结
  16. 青岛方言发音对照表(内附英文释义)
  17. 最新ThinkPHP微信独立精彩互换抢红包系统源码开源版
  18. 安装office的ISO版本,虚拟光驱
  19. STM32单片机蓝牙APP手势语音温控电风扇落地扇人体感应
  20. Tobii5的反复无常

热门文章

  1. dbm和mysql使用场景_mysql基本用法总结
  2. android wifi热点项目总结,高通Android wifi移植和wifi热点问题总结
  3. 文件服务器 权限和安全,NTFS安全权限、文件共享服务器
  4. linux sed 冒号,linux sed 总结
  5. mvc 扫描二维码跳转内部指定页面_开源神器:一个二维码,让文件传输不设限!...
  6. 李航《统计学习方法》之HMM隐马尔可夫模型
  7. 灰色关联法 —— matlab
  8. 没有bug队——加贝——Python 练习实例 1,2
  9. idea查看一个类的所有子类_java new一个对象的过程中发生了什么
  10. 无法执行二进制文件: 可执行文件格式错误_VB中.frm .frx .vbp .vbw .scc文件都有什么作用,你知道么?...