1. 程式人生 > >MFC筆記——GetSafeHwnd()和GetSafeHandle()的主要區別

MFC筆記——GetSafeHwnd()和GetSafeHandle()的主要區別

GetSafeHwnd()GetSafeHandle()的主要區別:

1.使用者不同:

1)窗體使用:

GetSafeHwnd()用於獲取窗體的安全控制代碼(即HWND),有了HWND我們就可以方便的對HWND指向的窗體進行所需的操作了;

2GDI物件使用:

GetSafeHandle(),用於獲取GDI物件的控制代碼。


注意:在使用指標時強烈建議這麼做:

// pSomeWnd 為一個窗體的指標

if ( NULL != pSomeWnd && NULL != pSomeWnd->GetSafeHwnd())

{

  // do something.

   }

//////////////////////////////////////////////////////////////////////

補充知識:

記憶體控制代碼與指標的區別:

1.控制代碼其實就是指標,但是他和指標最大的不同是:給你一個指標,你可以通過這個指標做任何事情,也許是好事,也許是通過這個指標破壞記憶體,幹一些搗亂的事情。這個我想大家都會碰到過,因為亂用指標可能會導致程式崩潰。
控制代碼就沒有這個缺點,通過控制代碼,你只能幹一些windows允許你乾的事情(例如呼叫一些api函式等等),沒有了指標的隨意。

2.控制代碼是一些表的索引也就是指向指標的指標。 控制代碼和指標都是地址,控制代碼是windows

程式設計的一個關鍵性的概念,編寫windows應用程式總是要和各種控制代碼打交道。
所謂控制代碼,就是一個唯一的數,用以標識許多不同的物件型別,如視窗、選單、記憶體、畫筆、畫刷等。在win32裡,控制代碼是指向一個無型別物件”(void)的指標,也就是一個4位元組長的資料。
無論它的本質是什麼,控制代碼並不是一個真正意義上的指標
從構造上看,控制代碼是一個指標,儘管它沒有指向用於儲存某個物件的記憶體位置。事實上,控制代碼指向一個包含了對該物件進行引用的位置。
控制代碼的宣告是這樣的:
typedef void handle

由於windows是一個多工作業系統,它可以同時執行多個程式或一個程式的多個副本。這些執行的程式稱為一個例項。為了對同一程式的多個副本進行管理,windows

引入了例項控制代碼。windows為每個應用程式建立一張表,例項控制代碼就好象是這張表的一個索引。

 
不同之處還在於:
1
、控制代碼所指的可以是一個很複雜的結構,並且很有可以是與系統有關的,比如說上面所說的執行緒的控制代碼,它指向的就是一個類或者結構,他和系統有很密切的關係,當一個執行緒由於不可預料的原因而終止時,系統就可以回收它所佔用的資源,如cpu,記憶體等等。反過來想可以知道,這個控制代碼中的某一些項,是與系統進行互動的。由於windows系統是一個多工的系統,它隨時都可能要分配記憶體,回收記憶體,重組記憶體等。 
2
、指標它也可以指向一個複雜的結構,但是通常是使用者定義的,所有的必需的工作都需使用者完成,特別是在刪除的時候。 但在vc++6.0中也有一些指標,它們都是在處理一些小問題才用的,如最常見的字元的指標 它也是需要使用者處理的,譬如你動態分配了記憶體;但是cstring就不要使用者處理了,它其實是vc++中的一個類,所有的操作都由成員函式完成,產生(分配)由建構函式,刪除(回收)由解構函式完成。


附註:獲得視窗控制代碼三種方法

1.HWND FindWindow( LPCTSTR lpClassName, LPCTSTR lpWindowName);

 

  HWND FindWindowEx( HWND hwndParent, HWND hwndChildAfter,
                        LPCTSTR lpszClass, LPCTSTR lpszWindow );

2.HWND WindowFromPoint( 
          POINT Point);//
獲得當前滑鼠游標位置的視窗hwnd

3.BOOL CALLBACK EnumChildProc( HWND hwnd, LPARAM lParam);

 

BOOL EnumChildWindows( HWND hWndParent,
                       WNDENUMPROC lpEnumFunc,
                       LPARAM lParam );
BOOL EnumWindows( WNDENUMPROC lpEnumFunc, LPARAM lParam ); 
BOOL CALLBACK EnumWindowsProc( HWND hwnd, LPARAM lParam );
 

補充知識:指標和控制代碼之間的轉換

a.由指標獲得控制代碼 
      CWND* pwnd ;
      HWND  hwnd ;
      hwnd = pwnd-> GetSafeHwnd();

b.由控制代碼得到指標:
      CWND* pwnd = FromeHandle(hmyhandle);
      pwnd-> SetWindowText(" hello world!" ) ;

mfc類中有的還提供了標準方法,比如window 控制代碼 : 
static CWND pascal FromHandle( HWND hwnd )
HWND GetSafeHwnd( ) const

對於點陣圖: 
static cbitmap pascal fromhandle( hbitmap hbitmap )
static cgdiobject pascal fromhandle( hgdiobj hobject )
hgdiobj getsafehandle( ) const


有人說控制代碼就是一個標示,一個id號,是錯誤的。一個id號可以包括多個資源,比如說單文件中的idr_mainframe,一般是指在硬碟上的資源。但是當把硬碟上的資源調入記憶體以後,將有一個控制代碼指向它,但是控制代碼只能指向一個資源。而且控制代碼知道所指的記憶體有多大。而指標指向地址,它不知道分配的記憶體有多大。
但是如果你定義一個控制代碼,然後在vc裡面右擊滑鼠,選擇" go to definition of handle,你會發現它的本質就是一個指標,但是它的作用不同於指標。

控制代碼是個指標,指向一塊記憶體,但至於這塊記憶體跟控制代碼所標識的物件是怎麼聯絡起來的,呼叫者不需要清楚,呼叫者只需要知道,這個控制代碼聯絡著一個win32物件。
    控制代碼是實體地址,可以跨程序傳遞,例如,handle ha程序a的一個視窗,你可以在程序b中利用一個跟ha相等的值(相等就是說它們強制轉成int32的值相等)初始化一個控制代碼,利用這個控制代碼你可以對程序a的那個物件進行操作,例如movewindow showwindow等。
控制代碼包含了一些引用計數之類的東西,所以我的上一點說的給控制代碼賦值是不安全的,windows api提供了一些函式,可以對控制代碼進行操作。


    控制代碼就是受限的指標。
它是由作業系統管理的,你不能通過它直接存取作業系統建立的資料結構(應該先獲取對應的指標)。

作業系統在建立一個物件(如gdi, file)等的時候,它會為這個物件上下文context保留一塊資料結構,然後把它放在一張全域性表中。控制代碼就是這塊資料結構在表中的索引。


原文來源:http://blog.sina.com.cn/s/blog_97312deb01015s3i.html