原文出自:http://blog.csdn.net/yuvmen/article/details/5877271
了解MFC程序的启动过程,对于初学者来讲,了学习MFC很有帮助;对于不常用VC的人来说,过一段时间就会忘记。还是来记下来,方便以后查阅。
1、创建Application object对象theApp
程序一开始生产一个(且只有一个)Application object对象theApp,也即一个CWinApp对象,这个全局对象一产生,便执行其构造函数,因为并没有定义CMyWinApp构造函数,所以即执行CWinApp类的构造函数。该函数定义于APPCORE.CPP第75行,你可以自己搜出来啃一啃,因此,CWinApp之中的成员变量将因为theApp这个全局对象的诞生而获得配置与初值。
2、WinMain登场
用SDK编程序时,程序的入口点是WinMain函数,而在MFC程序里我们并没有看到WinMain函数,哦!~ 原来她是被隐藏在MFC代码里面了。当theApp配置完成后,WinMain登场,慢!细看程序,并没连到WinMain函数的代码啊!这个我也不知道,MFC早已准备好并由链接器直接加到应用程序代码中了,原来她在APPMODUL.CPP里面,好,我们就认为当theApp配置完成后,程序就转到APPMODUL.CPP来了。那执行什么呢?看看下面从APPMODUL.CPP摘出来的代码:
extern "C" int WINAPI
_tWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)
{
// call shared/exported WinMain
return AfxWinMain(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
}
_tWinMain函数的“_t”是为了支持Unicode而准备的一个宏。
_tWinMain函数返回值是AfxWinMain函数的返回值,AfxWinMain函数定义于WINMAIN.CPP第21行,稍加整理,去芜存菁,就可以看到这个“程序进入点”主要做些什么事:
int AFXAPI AfxWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance, LPTSTR lpCmdLine, int nCmdShow)
{
int nReturnCode = -1;
CWinApp* pApp = AfxGetApp();
AfxWinInit(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
pApp->InitApplication();
pApp->InitInstance()
nReturnCode = pApp->Run();
AfxWinTerm();
return nReturnCode;
}
AfxGetApp()函数是取得CMyWinApp对象指针,故上面函数第6至8行相当于调用:
CMyWinApp::InitApplication();
CMyWinApp::InitInstance()
CMyWinApp::Run();
因而导致调用:
CWinApp::InitApplication(); //因为 CMyWinApp 并没有改写 InitApplication
CMyWinApp::InitInstance() //因为 CMyWinApp 改写了 InitInstance
CWinApp::Run(); //因为 CMyWinApp 并没有改写 Run
用过SDK写程序的朋友,现在可能会发出会心的微笑。
3、AfxWinInit——AFX内部初始化操作
AfxWinInit是继CWinApp构造函数之后的第一个操作,主要做的是AFX内部初始化操作,该函数定义于APPINIT.CPP第24行,这里就不掏出来了,你自己搜出来啃吧!
4、执行CWinApp::InitApplication
AfxWinInit之后的操作是pApp->InitApplication,我们已知道pApp指向CMyWinApp对象,当调用:
pApp->InitApplication();
相当于调用:
CMyWinApp::InitApplication();
但是你要知道,CMyWinApp继承自CWinApp,而InitApplication又是CWinApp的一个虚拟函数,我们并没有改写它(大部分情况下不需改写它),所以上述操作相当于调用:
CWinApp::InitApplication();
此函数定义于APPCORE.CPP第125行,你自己搜出来看吧!我就不搬出来了,里面的操作都是MFC为了内部管理而做的(其实我也看不懂,知道有这回事就好了)。
5、执行CWinApp::InitInstance
继InitApplication函数之后,AfxWinMain调用pApp->InitInstance。当程序调用:
pApp->InitInstance();
相当于调用:
CMyWinApp::InitInstance();
但是你要知道,CMyWinApp继承自CWinApp,而InitInstance又是CWinApp的一个虚拟函数。由于我们改写了它,所以上述操作就是调用我们自己(CMyWinApp)的这个InitInstance函数。
6、CFrameWnd::Create产生主窗口(并先注册窗口类)
现在已经来到CWinApp::InitInstance了,该函数先new一个CMyFrameWnd对象,从而产生主窗口。在创建CMyFrameWnd对之前,要先执行构造函数CMyFrameWnd::CMyFrameWnd(),该函数用Create函数产生窗口:
CMyFrameWnd::CMyFrameWnd()
{
Create(NULL, "Hello MFC", WS_OVERLAPPEDWINDOW, rectDefault, NULL, "MainMenu");
}
其中Create是CFrameWnd的成员函数,它将产生一个窗口,用过SDK编程序的朋友都知道,要创建主窗口时要先注册一个窗口类,规定窗口的属性等,但,这里使用哪一个窗口类呢?Create函数第一个参数(其它参数请参考MSDN或《深出浅出MFC》详解)指定窗口类设为NULL又是什么意思啊?意思是要以MFC内建的空中类产生一个标准的外框窗口,但,我们的程序一般都没有注册任何窗口类呀!噢,Create函数在产生窗口之前会引发窗口类的注册操作。
让我们先挖出Create函数都做了些什么操作,Create函数定义于WINFRM.CPP的第538行(在此我就不把代码Copy过来了,你自己打开出来看吧),函数在562行调用CreateEx函数,由于CreateEx是CWnd的成员函数,而CFrameWnd是从CWnd继而来,故将调用CWnd::CreateEx。此函数定义于WINCORE.CPP第665行,下面是部分代码:
BOOL CWnd::CreateEx(DWORD dwExStyle, LPCTSTR lpszClassName,
LPCTSTR lpszWindowName, DWORD dwStyle,
int x, int y, int nWidth, int nHeight,
HWND hWndParent, HMENU nIDorHMenu, LPVOID lpParam)
{
// allow modification of several common create parameters
CREATESTRUCT cs;
cs.dwExStyle = dwExStyle;
cs.lpszClass = lpszClassName;
cs.lpszName = lpszWindowName;
cs.style = dwStyle;
cs.x = x;
cs.y = y;
cs.cx = nWidth;
cs.cy = nHeight;
cs.hwndParent = hWndParent;
cs.hMenu = nIDorHMenu;
cs.hInstance = AfxGetInstanceHandle();
cs.lpCreateParams = lpParam;
if(PreCreateWindow(cs))
{
PostNcDestroy();
return FALSE;
}
AfxHookWindowCreate(this);
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);
...
}
用过SDK编程序的朋友,看到上面代码应该有一点感觉了吧,函数中调用的PreCreateWindows是虚拟函数,在CWnd和CFrameWnd之中都有定义。由于this指针所指对象的缘故,这里应该调用的是CFrameWnd::PreCreateWindow。该函数定义于WINFRM.CPP第521行,以下是部分代码:
BOOL CFrameWnd::PreCreateWindow(CREATESTRUCT& cs)
{
if (cs.lpszClass == NULL)
{
VERIFY(AfxDeferRegisterClass(AFX_WNDFRAMEORVIEW_REG));
cs.lpszClass = _afxWndFrameOrView; // COLOR_WINDOW background
}
...
}
其中AfxDeferRegisterClass是一个定义于AFXIMPL.H中的宏。该宏如下:
#define AfxDeferRegisterClass(fClass) AfxEndDeferRegisterClass(fClass)
注:这里有宏和《深入浅出MFC》的不一样,以上代码是从Visual C++ 6.0摘取。
AfxEndDeferRegisterClass定义于WINCORE.CPP第3619行,该函数很复杂,主要是注册五个窗口类(哇!终于看到窗口类了,怎么用5个呢?我还不清楚),不同类的PreCreateWindow成员函数都是在窗口产生之前一刻被调用,准备用来注册窗口类。如果我们指定的窗口类是NULL,那么就使用系统默认类。从CWnd及其各个派生类的PreCreateWindow成员函数可以看出,整个Framework针对不同功能的窗口使用了哪些窗口类。
7、窗口显示与更新
CMyFrameWnd::CMyFrameWnd结束后,窗口已经诞生出来;程序流程又回到CMyWinApp::InitInstance,于是调用ShowWindow函数令窗口显示出来,并调用UpdateWindow函数令程序送出WM_PAINT消息。在SDK程序中,消息是通过窗口函数来处理,而现在窗口函数在哪里、又如何送到窗口函数手中呢?那要从CWinApp::Run说起了。
8、执行CWinApp::Run——程序生命的活水源头
在执行完CMyWinApp::InitInstance函数后,程序的脚步到了AfxWinMain函数的pApp->Run了,现在我们已知道这将执行CWinApp::Run函数,该函数定义于APPCORE.CPP第391行,下面是程序代码:
int CWinApp::Run()
{
if (m_pMainWnd == NULL && AfxOleGetUserCtrl())
{
// Not launched /Embedding or /Automation, but has no main window!
TRACE0("Warning: m_pMainWnd is NULL in CWinApp::Run - quitting application./n");
AfxPostQuitMessage(0);
}
return CWinThread::Run();
}
函数调用CWinThread::Run函数,该函数定义于THRDCORE.CPP第456行,在这里我就不Copy出来了。函数在第480行调用了PumpMessage函数,该函数定义于THRDCORE.CPP第810行,整理后的部分代码如下:
BOOL CWinThread::PumpMessage()
{
if (!::GetMessage(&m_msgCur, NULL, NULL, NULL))
{
return FALSE;
}
// process this message
if (m_msgCur.message != WM_KICKIDLE && !PreTranslateMessage(&m_msgCur))
{
::TranslateMessage(&m_msgCur);
::DispatchMessage(&m_msgCur);
}
return TRUE;
}
该函数主要操作是将消息由::DispatchMessage送到窗口函数(CWnd::DefWindowProc)中,但程序一般没有提供任何窗口函数,但在AfxEndDeferRegisterClass中,在注册五种窗口类之前已经指定窗口函数为:
wndcls.lpfnWndProc = DefWindowProc;
虽然窗口函数被指定为DefWindowProc成员函数(CWnd::DefWindowProc),但事实上消息并不是被唧往该处,而是一个名为AfxWndProc的全局函数去。
9、把消息与处理函数连接在一起——Message Map机制
到此,主窗口已经产生,等待的就是各种消息了,然后调用相应的处理函数,然而消息和处理函数怎样连接在一起呢?MFC采用了Message Map机制(消息映射机制),提供给应用程序使用的“很方便的接口”的两组宏,其原理我还不大清楚,在这里也无法讲解,主要用法是:先在类声明中结合DECLARE_MESSAGE_MAP()给出处理函数,如:
class CMyFrameWnd : public CFrameWnd
{
public:
CMyFrameWnd();
afx_msg void OnPaint(); // for WM_PAINT
afx_msg void OnAbout(); // for WM_COMMAND (IDM_ABOUT)
DECLARE_MESSAGE_MAP()
}
再在相应的.CPP文件的任何位置(当然不能在函数之内)使用BEBIN_MESSAGE_MAP()和END_MESSAGE_MAP()宏把相应的消息加入去,如:
BEGIN_MESSAGE_MAP(CMyFrameWnd, CFrameWnd)
ON_COMMAND(IDM_ABOUT, OnAbout)
ON_WM_PAINT()
END_MESSAGE_MAP()
总结一下:
(1) 调用CXXApp构造函数构造全局对象theApp: CXApp theApp;
(2) 调用Winmain函数完成初始化工作: 通过宏_tWinMain
(3) 初始化工作包括: 窗口类注册、窗口产生、显示和更新、消息循环等等
① 注册窗口类:AfxEndDeferRegisterClass() //相当于SDK里面的RegisterClass()函数
② 创建窗口:CMainFrame::PreCreateWindow() //允许我们修改窗口属性的地方
CFrameWnd::Create()
③ 消息循环:PumpMessage()
在程序运行时CWinApp将创建一个CFrameWnd框架窗口实例,而框架窗口将创建文档模板,然后有文档模板创建文档实例和视实例,并将两者关联。
相关推荐
`theApp`的初始化过程是MFC程序启动的第一步,主要通过调用`__initterm()`函数来完成。这个函数会执行一系列的初始化操作,包括但不限于本地变量的初始化、运行时库的初始化等。 #### 1.2 theApp执行流程的深入剖析...
通过以上分析可以看出,MFC程序的执行流程与传统C语言程序有着本质的不同,但也遵循着一定的逻辑顺序。理解这些基本概念有助于开发者更好地掌握MFC的应用开发技巧,从而能够高效地开发出高质量的应用程序。
这个对象的构造函数在程序启动时自动调用,初始化一些成员变量,为后续的程序运行做好准备。如果你没有自定义`CMyWinApp`构造函数,那么会默认调用`CWinApp`的构造函数。 2. **WinMain函数的隐藏与调用** 传统的...
- **DoStartupMessageLoop**:接着,MFC会调用CWinApp::PreMessageLoop和CWinApp::OnIdle,这两个函数可以用来处理应用程序启动时的特殊任务。 - **主窗口创建**:应用程序的主要窗口,通常是CFrameWnd或...
这个函数在MFC程序的每个实例启动时都会被调用。它的主要职责是创建主窗口(顶级窗口)并设置程序运行的基本状态。`CreateWindow`函数在这个过程中扮演了关键角色,它根据之前注册的窗口类创建一个实际的窗口。`...
在深入探讨MFC SDI(单文档界面)应用程序启动过程中的关键步骤之前,我们需要对MFC框架有一个基本的理解。MFC(Microsoft Foundation Classes)是微软为简化Windows编程而提供的一套类库,它为开发者提供了许多封装...
### 全面解析MFC应用程序中处理消息的顺序 #### MFC消息处理流程概述 Microsoft Foundation Classes (MFC) 是一套广泛应用于Windows平台上的应用程序开发框架。为了更好地理解和掌握MFC应用程序的消息处理机制,本...
例如,`CWinApp`类的`InitInstance()`方法是初始化过程的核心,它会在应用程序启动时被调用,执行一系列的初始化操作。 接下来,`APPCORE.CPP`文件可能包含了应用程序的核心逻辑。在这个文件中,你可能会看到`...
1. 应用程序对象创建:在MFC程序启动时,首先会创建一个CWinApp派生类的对象,这是整个应用程序的入口点。CWinApp对象负责处理命令行参数、初始化COM库、注册窗口类等。 2. 主框架窗口创建:接着,应用程序会创建一...
6、MFC程序具有一个CTestApp类的全局对象theApp,在MFC程序运行时,程序执行的顺序为:theApp全局对象定义 处->CTestApp构造函数->WinMain函数 7、对于普通的VC++控制台程序,无论全局变量还是全局对象,程序运行时...
在这个“MFC程序的生命期应用实例源码”中,我们可以深入探讨MFC类的生命周期以及相关函数的作用。 MFC主要基于C++构建,它封装了Windows API,使得开发者可以更高效地编写Windows应用。MFC的主要类之一是CWinApp,...
而“mfc鼠标记录与回放程序”是利用MFC库实现的一种特殊应用,它的主要功能是对用户的鼠标动作进行记录,然后可以按照预设的顺序和速度回放这些动作,这种技术在自动化测试、演示脚本制作等方面非常有用。...
挂起操作通常用于控制线程的执行顺序或者处理线程间的同步问题。 - **启动线程**:当线程被挂起后,可以使用特定的API来恢复其执行,即启动线程。启动后,线程会从上次挂起的地方继续执行,直到完成或再次被挂起。 ...
- **初始化过程**:当MFC应用程序启动时,首先会调用`CWinApp`类的构造函数,接着是`InitInstance()`方法。在这里,开发者可以进行全局设置、资源加载等操作。 - **创建主窗口**:`InitInstance()`中通常会创建主...
由于Java Applet与MFC程序使用的字节顺序不同(前者为Big-Endian,后者为Little-Endian),需要在数据交换过程中进行转换。这一过程由`Swap`函数完成。例如,数字“4”在Big-Endian格式中表示为“0004”,而在Little...
在MFC应用程序启动时,执行顺序大致如下: 1. 应用程序入口点(WinMain或wWinMain)调用CWinApp的InitInstance。 2. CWinApp::InitInstance初始化MFC,包括加载资源、注册类、创建主窗口等。 3. 创建主框架窗口...
**MFC程序设计** MFC(Microsoft Foundation Classes)是微软为Windows平台开发的应用程序提供的一套C++类库,它基于面向对象编程理念,简化了Windows API的使用,使得开发者可以更加高效地构建Windows应用程序。...
在数据传输过程中,TCP提供顺序传输和错误检测,通过确认机制确保数据无丢失或重复。最后,通过四次挥手断开连接,释放资源。 在MFC框架下实现TCP通信,我们主要涉及以下关键步骤: 1. **创建MFC应用程序**:首先...