MFC Object和Windows Object的關系
MFC中最重要的封裝是對Win32 API的封裝,因此,理解Windows Object和MFC Object (C++對象,一個C++類的實例)之間的關系是理解MFC的關鍵之一。所謂Windows Object(Windows對象)是Win32下用句柄表示的Windows操作系統對象;所謂MFC Object (MFC對象)是C++對象,是一個C++類的實例,這裡(本書范圍內)MFC Object是有特定含義的,指封裝Windows Object的C++ Object,並非指任意的C++ Object。
MFC Object 和Windows Object是不一樣的,但兩者緊密聯系。以窗口對象為例:
一個MFC窗口對象是一個C++ CWnd類(或派生類)的實例,是程序直接創建的。在程序執行中它隨著窗口類構造函數的調用而生成,隨著析構函數的調用而消失。而Windows窗口則是Windows系統的一個內部數據結構的實例,由一個“窗口句柄”標識,Windows系統創建它並給它分配系統資源。Windows窗口在MFC窗口對象創建之後,由CWnd類的Create成員函數創建,“窗口句柄”保存在窗口對象的m_hWnd成員變量中。Windows窗口可以被一個程序銷毀,也可以被用戶的動作銷毀。MFC窗口對象和Windows窗口對象的關系如圖2-1所示。其他的Windows Object和對應的MFC Object也有類似的關系。
下面,對MFC Object和Windows Object作一個比較。有些論斷對設備描述表(MFC類是CDC,句柄是HDC)可能不適用,但具體涉及到時會指出。
從數據結構上比較
MFC Object是相應C++類的實例,這些類是MFC或者程序員定義的;
Windows Object是Windows系統的內部結構,通過一個句柄來引用;
MFC給這些類定義了一個成員變量來保存MFC Object對應的Windows Object的句柄。對於設備描述表CDC類,將保存兩個HDC句柄。
從層次上講比較
MFC Object是高層的,Windows Object是低層的;
MFC Object封裝了Windows Object的大部分或全部功能,MFC Object的使用者不需要直接應用Windows Object的HANDLE(句柄)使用Win32 API,代替它的是引用相應的MFC Object的成員函數。
從創建上比較
MFC Object通過構造函數由程序直接創建;Windows Object由相應的SDK函數創建。
MFC中,使用這些MFC Object,一般分兩步:
首先,創建一個MFC Object,或者在STACK中創建,或者在HEAP中創建,這時,MFC Object的句柄實例變量為空,或者說不是一個有效的句柄。
然後,調用MFC Object的成員函數創建相應的Windows Object,MFC的句柄變量存儲一個有效句柄。
CDC(設備描述表類)的創建有所不同,在後面的2.3節會具體說明CDC及其派生類的創建和使用。
當然,可以在MFC Object的構造函數中創建相應的Windows對象,MFC的GDI類就是如此實現的,但從實質上講,MFC Object的創建和Windows Object的創建是兩回事。
從轉換上比較
可以從一個MFC Object得到對應的Windows Object的句柄;一般使用MFC Object的成員函數GetSafeHandle得到對應的句柄。
可以從一個已存在的Windows Object創建一個對應的MFC Object; 一般使用MFC Object的成員函數Attach或者FromHandle來創建,前者得到一個永久性對象,後者得到的可能是一個臨時對象。
從使用范圍上比較
MFC Object對系統的其他進程來說是不可見、不可用的;而Windows Object一旦創建,其句柄是整個Windows系統全局的。一些句柄可以被其他進程使用。典型地,一個進程可以獲得另一進程的窗口句柄,並給該窗口發送消息。
對同一個進程的線程來說,只可以使用本線程創建的MFC Object,不能使用其他線程的MFC Object。
從銷毀上比較
MFC Object隨著析構函數的調用而消失;但Windows Object必須由相應的Windows系統函數銷毀。
設備描述表CDC類的對象有所不同,它對應的HDC句柄對象可能不是被銷毀,而是被釋放。
當然,可以在MFC Object的析構函數中完成Windows Object的銷毀,MFC Object的GDI類等就是如此實現的,但是,應該看到:兩者的銷毀是不同的。
每類Windows Object都有對應的MFC Object,下面用表格的形式列出它們之間的對應關系,如表2-1所示:
表2-1 MFC Object和Windows Object的對應關系
描述
Windows句柄
MFC Object
窗口
HWND
CWnd and CWnd-derived classes
設備上下文
HDC
CDC and CDC-derived classes
菜單
HMENU
CMenu
筆
HPEN
CGdiObject類,CPen和CPen-derived classes
刷子
HBRUSH
CGdiObject類,CBrush和CBrush-derived classes
字體
HFONT
CGdiObject類,CFont和CFont-derived classes
位圖
HBITMAP
CGdiObject類,CBitmap和CBitmap-derived classes
調色板
HPALETTE
CGdiObject類,CPalette和CPalette-derived classes
區域
HRGN
CGdiObject類,CRgn和CRgn-derived classes
圖像列表
HimageLIST
CimageList和CimageList-derived classes
套接字
SOCKET
CSocket,CAsynSocket及其派生類
表2-1中的OBJECT分以下幾類:
Windows對象,
設備上下文對象,
GDI對象(BITMAP,BRUSH,FONT,PALETTE,PEN,RGN),
菜單,
圖像列表,
網絡套接字接口。
從廣義上來看,文檔對象和文件可以看作一對MFC Object和Windows Object,分別用CDocument類和文件句柄描述。
後續幾節分別對前四類作一個簡明扼要的論述。
Windows Object
用SDK的Win32 API編寫各種Windows應用程序,有其共同的規律:首先是編寫WinMain函數,編寫處理消息和事件的窗口過程WndProc,在WinMain裡頭注冊窗口(Register Window),創建窗口,然後開始應用程序的消息循環。
MFC應用程序也不例外,因為MFC是一個建立在SDK API基礎上的編程框架。對程序員來說所不同的是:一般情況下,MFC框架自動完成了Windows登記、創建等工作。
下面,簡要介紹MFC Window對Windows Window的封裝。
Windows的注冊
一個應用程序在創建某個類型的窗口前,必須首先注冊該“窗口類”(Windows Class)。注意,這裡不是C++類的類。Register Window把窗口過程、窗口類型以及其他類型信息和要登記的窗口類關聯起來。
“窗口類”的數據結構
“窗口類”是Windows系統的數據結構,可以把它理解為Windows系統的類型定義,而Windows窗口則是相應“窗口類”的實例。Windows使用一個結構來描述“窗口類”,其定義如下:
typedef struct _WNDCLASSEX {
UINT cbSize; //該結構的字節數
UINT style; //窗口類的風格
WNDPROC lpfnWndProc; //窗口過程
int cbClsExtra;
int cbWndExtra;
HANDLE hInstance; //該窗口類的窗口過程所屬的應用實例
HICON hIcon; //該窗口類所用的像標
HCURSOR hCursor; //該窗口類所用的光標
HBRUSH hbrBackground; //該窗口類所用的背景刷
LPCTSTR lpszMenuName; //該窗口類所用的菜單資源
LPCTSTR lpszClassName; //該窗口類的名稱
HICON hIconSm; //該窗口類所用的小像標
} WNDCLASSEX;
從“窗口類”的定義可以看出,它包含了一個窗口的重要信息,如窗口風格、窗口過程、顯示和繪制窗口所需要的信息,等等。關於窗口過程,將在後面消息映射等有關章節作詳細論述。
Windows系統在初始化時,會注冊(Register)一些全局的“窗口類”,例如通用控制窗口類。應用程序在創建自己的窗口時,首先必須注冊自己的窗口類。在MFC環境下,有幾種方法可以用來注冊“窗口類”,下面分別予以討論。
調用AfxRegisterClass注冊
AfxRegisterClass函數是MFC全局函數。AfxRegisterClass的函數原型:
BOOL AFXAPI AfxRegisterClass(WNDCLASS *lpWndClass);
參數lpWndClass是指向WNDCLASS結構的指針,表示一個“窗口類”。
首先,AfxRegisterClass檢查希望注冊的“窗口類”是否已經注冊,如果是則表示已注冊,返回TRUE,否則,繼續處理。
接著,調用::RegisterClass(lpWndClass)注冊窗口類;
然後,如果當前模塊是DLL模塊,則把注冊“窗口類”的名字加入到模塊狀態的域m_szUnregisterList中。該域是一個固定長度的緩沖區,依次存放模塊注冊的“窗口類”的名字(每個名字是以“