MFC-原理分析

[align=center][b][size=large]MFC-原理分析[/size][/b][/align]

MFC只留出API供我们调用,我们使用其接口比较简单,但其内部封装及其复杂,要想成为真正的MFC高手,还是得对其内部的封装有所了解,下面就以一个简单的例子,作为对MFC内部结构探讨的开始。


//HELLO.h
class CMyApp : public CWinApp
{
public:
virtual BOOL InitInstance ();
};

class CMainWindow : public CFrameWnd
{
public:
CMainWindow ();

protected:
afx_msg void OnPaint ();
DECLARE_MESSAGE_MAP ()
};

//HELLO.cpp
#include
#include "Hello.h"

CMyApp myApp;

/
// CMyApp member functions

BOOL CMyApp::InitInstance ()
{
m_pMainWnd = new CMainWindow;
m_pMainWnd->ShowWindow (m_nCmdShow);
m_pMainWnd->UpdateWindow ();
return TRUE;
}

/
// CMainWindow message map and member functions

BEGIN_MESSAGE_MAP (CMainWindow, CFrameWnd)
ON_WM_PAINT ()
END_MESSAGE_MAP ()

CMainWindow::CMainWindow ()
{
Create (NULL, _T ("The Hello Application"));
}

void CMainWindow::OnPaint ()
{
CPaintDC dc (this);

CRect rect;
GetClientRect (&rect);

dc.DrawText (_T ("Hello, MFC"), -1, &rect,
DT_SINGLELINE | DT_CENTER | DT_VCENTER);
}



这个程序及其简单,就是输出"Hello, MFC"。

一、程序的入口

这个程序怎样运行的呢?它即没有main,也没有winmain,以及其他的函数入口,那么它是如何运行,难道会从天而降一个入口函数?

其实这个程序的入口在这里:

_tWinMain->AfxWinMain

-->int AFXAPI AfxWinMain()//位于:WINMAIN.cpp
{
CWinApp* pApp=AfxGetApp();//pApp指向CMyWinApp.
AfxWinInit(...);
pApp->InitApplication();
pApp->InitInstance();
nReturnCode = pApp->Run();
return 0;
}
其中AfxGetApp();//pApp指向CMyWinApp.就是取得CMyWinApp对象,所以AfxWinMain中的
pApp->InitApplication();
pApp->InitInstance();
nReturnCode = pApp->Run();

就相当与:
CMyWinApp->InitApplication();
CMyWinApp->InitInstance();
nReturnCode = CMyWinApp->Run();

也就是:
CWinApp->InitApplication();//CMyWinApp并没有改写InitApplication()
CMyWinApp->InitInstance();//CMyWinApp改写了InitInstance()
nReturnCode = CWinApp->Run();//CMyWinApp并没有改写Run()

所以其入口函数的调用顺序为:
_tWinMain->AfxWinMain -> CWinApp->InitApplication();
CMyWinApp->InitInstance();
CWinApp->Run();

这样便进入到当前程序里。

二、如何产生窗口


BOOL CMyApp::InitInstance ()
{
m_pMainWnd = new CMainWindow;
m_pMainWnd->ShowWindow (m_nCmdShow);
m_pMainWnd->UpdateWindow ();
return TRUE;
}


1. m_pMainWnd = new CMainWindow;
其中m_pMainWnd 是CWnd对象指针:CWnd* m_pMainWnd; 在这里用 用户自定义窗口CMainWindow初始化,从而调用CMainWindow::CMainWindow()


CMainWindow::CMainWindow ()
{
Create (NULL, _T ("The Hello Application"));
}


Create (NULL, _T ("The Hello Application"));用来创建一个窗口。

其声明为:
virtual BOOL Create(
LPCTSTR lpszClassName,
LPCTSTR lpszWindowName,
DWORD dwStyle,
const RECT& rect,
CWnd* pParentWnd,
UINT nID,
CCreateContext* pContext = NULL
);

2.m_pMainWnd->ShowWindow (m_nCmdShow);

当Create (NULL, _T ("The Hello Application"));完成后,程序又回到m_pMainWnd->ShowWindow (m_nCmdShow);来显示窗口。

3.m_pMainWnd->UpdateWindow ();

m_pMainWnd->ShowWindow (m_nCmdShow);来显示窗口后调用m_pMainWnd->UpdateWindow ();来更新窗口。

三、消息的传递

窗口创建后如何发送消息,使得响应消息的函数得以调用,本例中响应消息的函数是OnPaint ()用来在窗口中显示"Hello, MFC"字符,那么这个消息如何传递?

在调用m_pMainWnd->UpdateWindow ()时会发送WM_PAINT消息,此消息有谁获得,又是怎样发送出去的?

MFC程序和SDK程序一样,有一个GetMessage/DispatchMessage循环,用来获得消息和发送消息。而且每一个窗口都有一个窗口函数,并以某种方式进行消息的判断和响应。

当m_pMainWnd->UpdateWindow ()发送WM_PAINT消息后,CWinApp->Run();用循环判断来接受消息,并DispatchMessage发送出去。

四、MessageMap消息映射图及其消息响应机制

当CWinApp->Run();把接受到的消息发送出去后,是如何找到相应的响应函数,这就用到了MessageMap消息映射图。


DECLARE_MESSAGE_MAP ()
BEGIN_MESSAGE_MAP (CMainWindow, CFrameWnd)
ON_WM_PAINT ()
END_MESSAGE_MAP ()
afx_msg void OnMyPaint ();


这些宏构成了复杂的消息映射网。

CWinApp->Run()把接受到的消息发送到CWnd::DefWindowProc中,然后CWnd::DefWindowProc将绕行消息映射表MessageMap,绕行的过程中,发现吻合的函数,于是调用相应的函数。此函数是通过BEGIN_MESSAGE_MAP和END_MESSAGE_MAP之间的宏建立的连接。

这个例子中m_pMainWnd->UpdateWindow ()发送WM_PAINT消息,而CWinApp->Run()把接受到的消息发送到CWnd::DefWindowProc中,在映射表中会有
#define ON_WM_PAINT() \
{ WM_PAINT, 0, 0, 0, AfxSig_vv, \
(AFX_PMSG)(AFX_PMSGW) \
(static_cast< void (AFX_MSG_CALL CWnd::*)(void) > ( &ThisClass :: OnPaint)) },

会把WM_PAINT和&ThisClass :: OnPaint联系在一起,那么void CMainWindow::OnPaint ()就会被调用。

你可能感兴趣的:(MFC,mfc)