VC++MFC程序中的WinMain函数(二)

VC++MFC程序中的WinMain函数(二),第1张

目录

为什么程序会首先调用CTestApp类的构造函数呢?

接上:VC++基于MFC的程序框架剖析(一)

接下:VC++基于MFC的程序框架剖析(三)CWinApp类的定义(afxwin.h*)


为什么程序会首先调用CTestApp类的构造函数呢?

原因:程序中定义了一个CTestApp类型的全局对象:theApp。

提示: MFC程序的全局变量都放置在类视图窗口中的“全局函数和变量”分支下,单击该分支即可看到程序当前所有的全局函数和变量。

双击某个全局变量,即可定位到该变量的定义处。

CTestApp定义文件(Test.h):

// CTestApp:
// 有关此类的实现,请参阅 Test.cpp
//

class CTestApp : public CWinApp
{
public:
	CTestApp() noexcept;


// 重写
public:
	virtual BOOL InitInstance();
	virtual int ExitInstance();

// 实现
	afx_msg void OnAppAbout();
	DECLARE_MESSAGE_MAP()
};

extern CTestApp theApp;

CTestApp构造函数,源文件(Test.cpp):

// CTestApp 构造

CTestApp::CTestApp() noexcept
{
	// 支持重新启动管理器
	m_dwRestartManagerSupportFlags = AFX_RESTART_MANAGER_SUPPORT_ALL_ASPECTS;
#ifdef _MANAGED
	// 如果应用程序是利用公共语言运行时支持(/clr)构建的,则: 
	//     1) 必须有此附加设置,“重新启动管理器”支持才能正常工作。

// 2) 在您的项目中,您必须按照生成顺序向 System.Windows.Forms 添加引用。

System::Windows::Forms::Application::SetUnhandledExceptionMode(System::Windows::Forms::UnhandledExceptionMode::ThrowException); #endif // TODO: 将以下应用程序 ID 字符串替换为唯一的 ID 字符串;建议的字符串格式 //为 CompanyName.ProductName.SubProduct.VersionInformation SetAppID(_T("Test.AppID.NoVersion")); // TODO: 在此处添加构造代码, // 将所有重要的初始化放置在 InitInstance 中 }

程序执行的顺序:theApp全局对象定义处、CTestApp构造函数、WinMain函数。

在程序入口main函数加载时,系统就已经为全局变量或全局对象分配了存储空间,并为它们赋了初始值。

小技巧: 在程序运行过程中,如果想要查看某个变量的当前值,那么方法一是把鼠标移到该变量上停留片刻,Visual Studio就会d出一个小窗口,在此窗口中显示了该变量的当前值,如图所示:

为什么全局变量theApp的构造函数会在WinMain函数之前执行?为什么要定义一个全局对象theApp,让它在WinMain函数之前执行?该对象的作用是什么?

应用程序的实例是由实例句柄(WinMain函数的参数hInstance)来标识的。

而对 MFC 程序来说,通过产生一个应用程序类的对象来唯一标识应用程序的实例。

每一个MFC程序有且仅有一个从应用程序类(CWinApp)派生的类。

每一个MFC程序实例有且仅有一个该派生类的实例化对象,也就是theApp全局对象。

该对象就表示了应用程序本身。

当一个子类在构造之前会先调用其父类的构造函数。

因此 theApp 对象的构造函数 CTestApp 在调用之前,会调用其父类CWinApp的构造函数,从而就把我们程序自己创建的类与Microsoft提供的基类关联起来了。

CWinApp的构造函数完成程序运行时的一些初始化工作。

CWinApp构造函数代码(appcore.cpp):

CWinApp::CWinApp(LPCTSTR lpszAppName)
{
	if (lpszAppName != NULL)
		m_pszAppName = _tcsdup(lpszAppName);
	else
		m_pszAppName = NULL;

	// initialize CWinThread state
	AFX_MODULE_STATE* pModuleState = _AFX_CMDTARGET_GETSTATE();
	ENSURE(pModuleState);
	AFX_MODULE_THREAD_STATE* pThreadState = pModuleState->m_thread;
	ENSURE(pThreadState);
	ASSERT(AfxGetThread() == NULL);
	pThreadState->m_pCurrentWinThread = this;
	ASSERT(AfxGetThread() == this);
	m_hThread = ::GetCurrentThread();
	m_nThreadID = ::GetCurrentThreadId();

	// initialize CWinApp state
	ASSERT(afxCurrentWinApp == NULL); // only one CWinApp object please
	pModuleState->m_pCurrentWinApp = this;
	ASSERT(AfxGetApp() == this);

	// in non-running state until WinMain
	m_hInstance = NULL;
	m_hLangResourceDLL = NULL;
	m_pszHelpFilePath = NULL;
	m_pszProfileName = NULL;
	m_pszRegistryKey = NULL;
	m_pszExeName = NULL;
	m_pszAppID = NULL;
	m_pRecentFileList = NULL;
	m_pDocManager = NULL;
	m_atomApp = m_atomSystemTopic = NULL;
	m_lpCmdLine = NULL;
	m_pCmdInfo = NULL;
	m_pDataRecoveryHandler = NULL;

	// initialize wait cursor state
	m_nWaitCursorCount = 0;
	m_hcurWaitCursorRestore = NULL;

	// initialize current printer state
	m_hDevMode = NULL;
	m_hDevNames = NULL;
	m_nNumPreviewPages = 0;     // not specified (defaults to 1)

	// initialize DAO state
	m_lpfnDaoTerm = NULL;   // will be set if AfxDaoInit called

	// other initialization
	m_bHelpMode = FALSE;
	m_eHelpType = afxWinHelp;
	m_nSafetyPoolSize = 512;        // default size

	m_dwRestartManagerSupportFlags = 0;    // don't support Restart Manager by default
	m_nAutosaveInterval = 5 * 60 * 1000;   // default autosave interval is 5 minutes (only has effect if autosave flag is set)

	m_bTaskbarInteractionEnabled = TRUE;

	// Detect the kind of OS:
	OSVERSIONINFO osvi;
	osvi.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);

// Fix for warnings when building against WinBlue build 9444.0.130614-1739
// warning C4996: 'GetVersionExW': was declared deprecated
// externalapis\windows\8.1\sdk\inc\sysinfoapi.h(442)
// Deprecated. Use VerifyVersionInfo* or IsWindows* macros from VersionHelpers.
#pragma warning( disable : 4996 )
	::GetVersionEx(&osvi);
#pragma warning( default : 4996 )

	m_bIsWindows7 = (osvi.dwMajorVersion == 6) && (osvi.dwMinorVersion >= 1) || (osvi.dwMajorVersion > 6);

	// Taskbar initialization:
	m_bComInitialized = FALSE;

	m_pTaskbarList = NULL;
	m_pTaskbarList3 = NULL;
	m_bTaskBarInterfacesAvailable = TRUE;
}

CWinApp的构造函数中有这样两行代码:

pThreadState->m_pCurrentWinThread = this;
pModuleState->m_pCurrentWinApp = this;

m_pCurrentWinThread对象的类型是CWinThread,该类是CWinApp的父类。

根据C++继承性原理,这个this对象代表的是子类CTestApp的对象,即theApp。

CWinApp的构造函数有一个LPCTSTR类型的形参:lpszAppName,但是CTestApp的构造函数没有参数。

如果基类的构造函数带有一个形参,那么子类构造函数需要显式地调用基类带参数的构造函数。

为什么我们程序中的CTestApp构造函数没有这么做呢?

如果某个函数的参数有默认值,那么在调用该函数时可以传递该参数的值,也可以不传递,直接使用默认值即可。

在CWinApp 类名上单击鼠标右键,利用【转到定义】命令,定位到CWinApp类的定义处,代码下:CWinApp构造函数的形参确实有一个默认值(NULL)。

这样,在调用CWinApp类的构造函数时,就不用显式地去传递这个参数的值。

class CWinApp : public CWinThread
{
	DECLARE_DYNAMIC(CWinApp)
public:

// Constructor
	explicit CWinApp(LPCTSTR lpszAppName = NULL);     // app name defaults to EXE name
...
}
接上:VC++基于MFC的程序框架剖析(一) 接下:VC++基于MFC的程序框架剖析(三)CWinApp类的定义(afxwin.h*)

欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/langs/674084.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2022-04-19
下一篇 2022-04-19

发表评论

登录后才能评论

评论列表(0条)

保存