明明白白看MFC之程序框架(二)

发表于:2007-07-01来源:作者:点击数: 标签:
二、 MFC应用程序结构 在《明明白白看MFC之程序框架(一)》中我分析了一个经典的SDK应用程序的结构,现在可是要进入主题“MFC应用程序结构”了。MFC应用程序有好多种,为了能够更清楚地与前面的文章形成对比,我们在这里看一个SDI的应用程序,当然例子还是


二、     MFC应用程序结构

在《明明白白看MFC之程序框架(一)》中我分析了一个经典的SDK应用程序的结构,现在可是要进入主题“MFC应用程序结构”了。MFC应用程序有好多种,为了能够更清楚地与前面的文章形成对比,我们在这里看一个SDI的应用程序,当然例子还是经典的“Hello World”了。在使用向导生成应用程序后,会发现有好几个文件,首先我们不管有哪些文件,按照程序执行得主线抽取主要的源程序分析一下再说(因为MFC生成的应用程序不是很方便阅读,所以在这里我将他们重新编辑了)。



CHelloWorldApp theApp;

int AFXAPI AfxWinMain(HINSTANCE hInstance, HINSTANCE hPrevInstance,

   LPTSTR lpCmdLine, int nCmdShow)

{

  

   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)

      {

          TRACE0("Warning: Destroying non-NULL m_pMainWnd\n");

          pThread->m_pMainWnd->DestroyWindow();

      }

      nReturnCode = pThread->ExitInstance();

      goto InitFailure;

   }

   nReturnCode = pThread->Run();



InitFailure:

……………………



   AfxWinTerm();

   return nReturnCode;

}



BOOL CWinApp::InitApplication()

{

   if (CDocManager::pStaticDocManager != NULL)

   {

      if (m_pDocManager == NULL)

          m_pDocManager = CDocManager::pStaticDocManager;

      CDocManager::pStaticDocManager = NULL;

   }



   if (m_pDocManager != NULL)

      m_pDocManager->AddDocTemplate(NULL);

   else

      CDocManager::bStaticInit = FALSE;



   return TRUE;

}

BOOL CHelloWorldApp::InitInstance()

{

   AfxEnableControlContainer();



   ………………………………



   // Change the registry key under which our settings are stored.

   // TODO: You should modify this string to be something appropriate

   // such as the name of your company or organization.

   SetRegistryKey(_T("Local AppWizard-Generated Applications"));



   LoadStdProfileSettings(); // Load standard INI file options (including MRU)



   // Register the application@#s document templates. Document templates

   // serve as the connection between documents, frame windows and views.



   CSingleDocTemplate* pDocTemplate;

   pDocTemplate = new CSingleDocTemplate(

      IDR_MAINFRAME,

      RUNTIME_CLASS(CHelloWorldDoc),

      RUNTIME_CLASS(CMainFrame),    // main SDI frame window

      RUNTIME_CLASS(CHelloWorldView));

   AddDocTemplate(pDocTemplate);



   // Parse command line for standard shell commands, DDE, file open

   CCommandLineInfo cmdInfo;

   ParseCommandLine(cmdInfo);



   // Dispatch commands specified on the command line

   if (!ProcessShellCommand(cmdInfo))

      return FALSE;



   // The one and only window has been initialized, so show and update it.

   m_pMainWnd->ShowWindow(SW_SHOW);

   m_pMainWnd->UpdateWindow();



   return TRUE;

}

BOOL CWinApp::InitInstance()

{

   return TRUE;

}

        

               MFC应用程序之“Hello World”



   咋一眼看上去,好像这个程序无从下手分析,甚至连程序的入口点都找不到。其实,上面的程序还是经过整理后才有如此模样。好了,一样的来看看这个程序是怎么运行的吧(要注意的事上面的程序来自于不同的文件,这里排版在一起只是为了更清楚地表示程序的结构,至于MFC的文件组织我会在下面一个话题中具体的分析,这里可以暂时不考虑)。

首先,在程序的开始处,首先定义了一个全局变量theApp,我们现在只需要知道他代表了整个程序的存在,然后程序开始介入入口点。有没有搞错,入口点在哪里?不及,其实int AFXAPI AfxWinMain()就是这个程序的入口点,奇怪吧!不过没有关系,就好像我们第一次看到C语言中的main()函数一样,只要了解就可以了。在AfxWinMain()中分别调用了一些类的成员函数,仿照前面的分析方法,也可以画出一个程序执行路径图。入口点----〉AfxGetThread()------〉AfxGetApp()-------àAfxWinInit()-------àpApp->InitApplication()-----àpThread->InitInstance()------àpThread->Run()。可以看到,程序一样有一个执行的线索可循,但是,相对于SDK来说,如今已经面目全非了,过去的那种清晰的程序结构在这些程序中也有吗?答案是肯定的,只不过他们的具体实现在MFC中都进行了包装而已,那么,还是来看看这个应用程序是如何启动并且运行的吧。

程序由AfxWinMain()开始运行后,首先调用了AfxGetApp()来获取应用程序的对象指针pApp,然后通过这个指针调用有关的成员函数来完成初始化和启动工作,最后就调用了Run()函数,在这里,Run()函数就是代表了SDK中的消息循环。

事情的发展在预料中进行着,但是似乎还遗漏了一点什么似的?不错,在上面我们的确是还有一样工作没有完成,这就是我们需要的”Hello World”好像还没有输出来!这不是我的疏忽,而是故意的安排,因为MFC中采用了一种全新(当然是相对于SDK来说的了)的消息处理机制,至少在表面上来说是这样的。然而,在这里我不打算一下子就把问题解决掉,毕竟这有点复杂,等我们明白了MFC的文件之间的关系后我会在回答这个问题。

我没有想到这篇文章会这么长,刚刚开始的时候我以为我可以一下子就把这个问题说清楚地,但是事实上我写作的思路也像我分析程序时一样,竟然是一个漫长的过程!所以我也有必要提醒一下自己以及读这篇文章的朋友,应该休息一下了。我现在都开始庆幸我把这个问题分成好几个小问题来解决了,我会在接下来的话题中继续讨论的事MFC程序所生成的文件以及它们之间的调用关系。(待续)

原文转自:http://www.ltesting.net