MFC是如何工作的?
MFC本质就是一个类库,它封装了Windows API的接口,你可以直接调用类来完成程序设计。
在你使用VS建立一个MFC项目时,VS已经贴心的为你准备好了两个头文件和两个主文件,让我们来看看吧。
第一步:建立MFC工程项目。
第二步:选择默认为你创建的基于对话框的MFC应用。
工程名头文件
进入工程后,你可以前往“解决方案资源管理器->头文件->工程项目名.h”找到这一头文件,我们先说这个文件。
可以看到这个头文件的主要目的就是以CWinAPP这个公共类为模板,扩展出一个新的专属于你的APP的类。
在实际执行EXE文件的过程中,根据你编译器的指令,会有不同的入口函数,一般的MFC程序,其真实的入口函数已经隐藏在其类库之中,所以你在写的时候,只需调用基本的CWinAPP类,他会有类函数作为初始入口函数的。
也就是说,当你使用MFC类的时候,没有C++传统的主函数,没有入口函数,你只能定义两个东西,一个是APP类,一个是Dialog类,前者是出一个应用程序进程,后者是这个进程中各组件、控件(对基于对话框的文件来说是如此的)。
有关MFC类库的信息,可以去这里查:https://learn.microsoft.com/zh-cn/cpp/mfc/mfc-desktop-applications?view=msvc-170。
主函数在哪里?
根据上面文档中的描述,直观的说,MFC没有main函数,也就不知道程序执行的顺序。可是,注意到只有一个工程名CPP文件,并且工程名文件“CMyApp”这个类过于耀眼,直觉告诉你,所谓的mian函数应该就在里面。
MFC的库类文件所在位置为:\VS\IDE\VC\Tools\MSVC\14.34.31933\atlmfc\src\mfc。在这一文件夹下的各类cpp文件就是MFC封装好的类。
class CMyApp : public CWinApp;
CMyApp theApp;
如上代码所示,我们真实的main函数应该就在这个类中,这里的CWinAPP父类里一定有我们理解中的主函数。打开D:\VS\IDE\VC\Tools\MSVC\14.34.31933\atlmfc\include\afxwin.h文件找到CWinApp定义文件,发现它又是CWinThread的子类,CWinThread又是CCmdTarget的子类。
当我们声明一个theApp的全局对象时,CWinApp之中的成员变量都会获得配置和初始值。
在\VS\IDE\VC\Tools\MSVC\14.34.31933\atlmfc\src\mfc位置中找到appmodul.cpp文件。里面会有winmain的声明。
extern "C" int WINAPI
_tWinMain(_In_ HINSTANCE hInstance, _In_opt_ HINSTANCE hPrevInstance,
_In_ LPTSTR lpCmdLine, _In_ int nCmdShow)
#pragma warning(suppress: 4985)
{
// call shared/exported WinMain
return AfxWinMain(hInstance, hPrevInstance, lpCmdLine, nCmdShow);
}
这个就是MFC中内置的main函数了。(PS:如果有什么函数你找不到的话,可以去atlmfc下的src搜索)
在winmain.cpp中,AfxWinMain定义为:
int AFXAPI AfxWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,
_In_ LPTSTR lpCmdLine, int nCmdShow)
{
ASSERT(hPrevInstance == NULL);
int nReturnCode = -1;
CWinThread* pThread = AfxGetThread();
CWinApp* pApp = AfxGetApp();
// AFX internal initialization
if (!AfxWinInit(hInstance, hPrevInstance, lpCmdLine, nCmdShow))
goto InitFailure;
// App global initializations (rare)
if (pApp != NULL && !pApp->InitApplication())
goto InitFailure;
// Perform specific initializations
if (!pThread->InitInstance())
{
if (pThread->m_pMainWnd != NULL)
{
TRACE(traceAppMsg, 0, "Warning: Destroying non-NULL m_pMainWnd\n");
pThread->m_pMainWnd->DestroyWindow();
}
nReturnCode = pThread->ExitInstance();
goto InitFailure;
}
nReturnCode = pThread->Run();
InitFailure:
#ifdef _DEBUG
// Check for missing AfxLockTempMap calls
if (AfxGetModuleThreadState()->m_nTempMapLock != 0)
{
TRACE(traceAppMsg, 0, "Warning: Temp map lock count non-zero (%ld).\n",
AfxGetModuleThreadState()->m_nTempMapLock);
}
AfxLockTempMaps();
AfxUnlockTempMaps(-1);
#endif
AfxWinTerm();
return nReturnCode;
}
其中,CWinApp* pApp = AfxGetApp();意味初始化APP指针。AfxWinInit完成了线程的初始化和窗框类的注册。CWinApp::InitApplication()完成了MFC的内容管理。
可以说,MFC类的最终执行依靠的时InitInstance函数。