考慮單窗口情況:
假設(shè)自己通過(guò)new創(chuàng)建了一個(gè)窗口對(duì)象pWnd,然后pWnd->Create。則銷毀窗口的調(diào)用次序:
1. 手工調(diào)用pWnd->DestroyWindow();
2. DestroyWindow會(huì)發(fā)送WM_DESTROY;
3. WM_DESTROY對(duì)應(yīng)的消息處理函數(shù)是OnDestroy();
4. DestroyWindow會(huì)發(fā)送WM_NCDESTROY;
5. WM_NCDESTROY對(duì)應(yīng)的消息處理函數(shù)是OnNcDestroy;
6. OnNcDestroy最后會(huì)調(diào)用PostNcDestroy;
7. PostNcDestroy經(jīng)常被用戶重載以提供釋放內(nèi)存操作。例如可以使用delete this;
通過(guò)這種方式,窗口對(duì)象對(duì)應(yīng)的窗口和窗口對(duì)象本身都被釋放了。
含有子窗口的情況
如果含有子窗口,則調(diào)用父窗口的DestroyWindow時(shí),它會(huì)向子窗口發(fā)送WM_DESTROY和WM_NCDESTROY消息。
具體調(diào)用順序參考下文的例子。
DestroyWindow對(duì)delete的影響:
應(yīng)該說(shuō)前者對(duì)后者并沒(méi)有什么影響。但經(jīng)常在DestroyWindow間接導(dǎo)致執(zhí)行的PostNcDestroy中delete窗口對(duì)象指針,即delete this。
CView::PostNcDestroy中唯一的操作就是delete this;CframeWnd::PostNcDestory也是如此。而默認(rèn)的CWnd::PostNcDestroy是空操作,CDialog中也沒(méi)有對(duì)其進(jìn)行重載,即也是空。
delete對(duì)Destroy的影響:
delete會(huì)導(dǎo)致析構(gòu)函數(shù)。CWnd的析構(gòu)函數(shù)中有對(duì)DestroyWindow的調(diào)用,但必須保證:
m_hWnd != NULL &&
this != (CWnd*) &wndTop &&this != (CWnd*)&wndBottom &&
this != (CWnd*)&wndTopMost &&this != (CWnd*)&wndNoTopMost。
Cdialog的析構(gòu)函數(shù)中也有對(duì)DestroyWindow的調(diào)用,但條件比較松,只需要m_hWnd != NULL。另外Cdialog::DoModal也會(huì)調(diào)用DestroyWindow。
CFrameWnd的OnClose中會(huì)調(diào)用DestroyWindow,但其析構(gòu)中不會(huì)調(diào)用DestroyWindow。
CView的析構(gòu)也不會(huì)調(diào)用DestroyWindow。
一個(gè)SDI程序的銷毀過(guò)程
有CMainFrame類、CMyView類。并且CMyView有兩個(gè)子窗口CMyDlg和CmyWnd的實(shí)例。
點(diǎn)擊退出按鈕,CMainFrame會(huì)收到WM_CLOSE消息。CframeWnd(CMainFrame的父類)間接會(huì)調(diào)用CWnd::DestroyWindow;它首先向CMyView發(fā)送WM_DESTORY和WM_NCDESTROY消息,并引發(fā)相應(yīng)的處理函數(shù);然后向CMyDlg發(fā)送WM_DESTORY和WM_NCDESTROY消息,并引發(fā)相應(yīng)的處理函數(shù);然后向CMyWnd發(fā)送WM_DESTORY和WM_NCDESTROY消息,并引發(fā)相應(yīng)的處理函數(shù)。
具體的執(zhí)行順序是:
1. 調(diào)用CMainFrame::DestroyWindow
2. CFrameWnd::OnDestroy
3. CMyView::OnDestroy
4. CmyWnd::OnDestroy
5. CmyDlg::OnDestroy
6. CmyWnd::PostNcDestroy
7. CmyWnd的析構(gòu)
8. CmyDlg::OnDestroy
9. CmyDlg的析構(gòu)
10. CMyView::PostNcDestroy
11. CmyView的析構(gòu)
12. CMainFrame的析構(gòu)
13. CMainFrame::DestroyWindow退出
上面情況是假設(shè)我們?cè)贑myWnd和CmyDlg的PostNcDestroy中添加了delete this。如果沒(méi)有添加,則7,10不會(huì)執(zhí)行。
因?yàn)镃View::PostNcDestroy中調(diào)用了delete this,所以然后會(huì)執(zhí)行CMyView的析構(gòu)操作。因?yàn)镃frameWnd::PostNcDestroy中調(diào)用了delete this,所以最后執(zhí)行CMainFrame的析構(gòu)操作。
如果自己的CmyDlg和CmyWnd在PostNcDestroy中有delete this;則二者會(huì)被析構(gòu)。否則內(nèi)存泄漏。當(dāng)然delete也可以放在CMyView的析構(gòu)中做,只是不夠OO。
總結(jié)
可以有兩種方法銷毀窗口對(duì)象對(duì)應(yīng)的窗口和釋放窗口對(duì)象指針。一種是通過(guò)DestroyWindow。這是比較好的方法,因?yàn)樽詈驧FC會(huì)自動(dòng)相應(yīng)WM_CLOSE導(dǎo)致CframWnd::DestroyWindow被調(diào)用,然后會(huì)一次釋放所有子窗口的句柄。用戶需要做的是在PostNcDestroy中釋放堆窗口對(duì)象指針。但因?yàn)槟承?duì)象是在棧中申請(qǐng)的,所以delete this可能出錯(cuò)。這就要保證寫(xiě)程序時(shí)自己創(chuàng)建的窗口盡量使用堆申請(qǐng)。
另一種是delete。Delete一個(gè)窗口對(duì)象指針有的窗口類(如CWnd,Cdialog)會(huì)間接調(diào)用DestroyWindow,有的窗口類(如CView,CframeWn)不會(huì)調(diào)用DestroyWindow。所以要小心應(yīng)對(duì)。
二者是相互調(diào)用的,很繁瑣。
一個(gè)MFC窗口對(duì)象包括兩方面的內(nèi)容:一是窗口對(duì)象封裝的窗口,即存放在m_hWnd成員中的HWND(窗口句柄),二是窗口對(duì)象本身是一個(gè)C++對(duì)象。要?jiǎng)h除一個(gè)MFC窗口對(duì)象,應(yīng)該先刪除窗口對(duì)象封裝的窗口,然后刪除窗口對(duì)象本身。
刪除窗口最直接方法是調(diào)用CWnd::DestroyWindow或::DestroyWindow,前者封裝了后者的功能。前者不僅會(huì)調(diào)用后者,而且會(huì)使成員m_hWnd保存的HWND無(wú)效(NULL)。如果DestroyWindow刪除的是一個(gè)父窗口或擁有者窗口,則該函數(shù)會(huì)先自動(dòng)刪除所有的子窗口或被擁有者,然后再刪除父窗口或擁有者。在一般情況下,在程序中不必直接調(diào)用DestroyWindow來(lái)刪除窗口,因?yàn)镸FC會(huì)自動(dòng)調(diào)用DestroyWindow來(lái)刪除窗口。例如,當(dāng)用戶退出應(yīng)用程序時(shí),會(huì)產(chǎn)生WM_CLOSE消息,該消息會(huì)導(dǎo)致MFC自動(dòng)調(diào)用CWnd::DestroyWindow來(lái)刪除主框架窗口,當(dāng)用戶在對(duì)話框內(nèi)按了OK或Cancel按鈕時(shí),MFC會(huì)自動(dòng)調(diào)用CWnd::DestroyWindow來(lái)刪除對(duì)話框及其控件。
窗口對(duì)象本身的刪除則根據(jù)對(duì)象創(chuàng)建方式的不同,分為兩種情況。在MFC編程中,會(huì)使用大量的窗口對(duì)象,有些窗口對(duì)象以變量的形式嵌入在別的對(duì)象內(nèi)或以局部變量的形式創(chuàng)建在堆棧上,有些則用new操作符創(chuàng)建在堆中。對(duì)于一個(gè)以變量形式創(chuàng)建的窗口對(duì)象,程序員不必關(guān)心它的刪除問(wèn)題,因?yàn)樵搶?duì)象的生命期總是有限的,若該對(duì)象是某個(gè)對(duì)象的成員變量,它會(huì)隨著父對(duì)象的消失而消失,若該對(duì)象是一個(gè)局部變量,那么它會(huì)在函數(shù)返回時(shí)被清除。
對(duì)于一個(gè)在堆中動(dòng)態(tài)創(chuàng)建的窗口對(duì)象,其生命期卻是任意長(zhǎng)的。初學(xué)者在學(xué)習(xí)C++編程時(shí),對(duì)new操作符的使用往往不太踏實(shí),因?yàn)橛胣ew在堆中創(chuàng)建對(duì)象,就不能忘記用delete刪除對(duì)象。讀者在學(xué)習(xí)MFC的例程時(shí),可能會(huì)產(chǎn)生這樣的疑問(wèn),為什么有些程序用new創(chuàng)建了一個(gè)窗口對(duì)象,卻未顯式的用delete來(lái)刪除它呢?問(wèn)題的答案就是有些MFC窗口對(duì)象具有自動(dòng)清除的功能。
如前面講述非模態(tài)對(duì)話框時(shí)所提到的,當(dāng)調(diào)用CWnd::DestroyWindow或::DestroyWindow刪除一個(gè)窗口時(shí),被刪除窗口的PostNcDestroy成員函數(shù)會(huì)被調(diào)用。缺省的PostNcDestroy什么也不干,但有些MFC窗口類會(huì)覆蓋該函數(shù)并在新版本的PostNcDestroy中調(diào)用delete this來(lái)刪除對(duì)象,從而具有了自動(dòng)清除的功能。此類窗口對(duì)象通常是用new操作符創(chuàng)建在堆中的,但程序員不必操心用delete操作符去刪除它們,因?yàn)橐坏┱{(diào)用DestroyWindow刪除窗口,對(duì)應(yīng)的窗口對(duì)象也會(huì)緊接著被刪除。
不具有自動(dòng)清除功能的窗口類如下所示。這些窗口對(duì)象通常是以變量的形式創(chuàng)建的,無(wú)需自動(dòng)清除功能。
所有標(biāo)準(zhǔn)的Windows控件類。
1. 從CWnd類直接派生出來(lái)的子窗口對(duì)象(如用戶定制的控件)。
2. 切分窗口類CSplitterWnd。
3. 缺省的控制條類(包括工具條、狀態(tài)條和對(duì)話條)。
4. 模態(tài)對(duì)話框類。
具有自動(dòng)清除功能的窗口類如下所示,這些窗口對(duì)象通常是在堆中創(chuàng)建的。
1. 主框架窗口類(直接或間接從CFrameWnd類派生)。
2. 視圖類(直接或間接從CView類派生)。
在設(shè)計(jì)自己的派生窗口類時(shí),可根據(jù)窗口對(duì)象的創(chuàng)建方法來(lái)決定是否將窗口類設(shè)計(jì)成可以自動(dòng)清除的。例如,對(duì)于一個(gè)非模態(tài)對(duì)話框來(lái)說(shuō),其對(duì)象是創(chuàng)建在堆中的,因此應(yīng)該具有自動(dòng)清除功能。
綜上所述,對(duì)于MFC窗口類及其派生類來(lái)說(shuō),在程序中一般不必顯式刪除窗口對(duì)象。也就是說(shuō),既不必調(diào)用DestroyWindow來(lái)刪除窗口對(duì)象封裝的窗口,也不必顯式地用delete操作符來(lái)刪除窗口對(duì)象本身。只要保證非自動(dòng)清除的窗口對(duì)象是以變量的形式創(chuàng)建的,自動(dòng)清除的窗口對(duì)象是在堆中創(chuàng)建的,MFC的運(yùn)行機(jī)制就可以保證窗口對(duì)象的徹底刪除。
如果需要手工刪除窗口對(duì)象,則應(yīng)該先調(diào)用相應(yīng)的函數(shù)(如CWnd::DestroyWindow)刪除窗口,然后再刪除窗口對(duì)象.對(duì)于以變量形式創(chuàng)建的窗口對(duì)象,窗口對(duì)象的刪除是框架自動(dòng)完成的.對(duì)于在堆中動(dòng)態(tài)創(chuàng)建了的非自動(dòng)清除的窗口對(duì)象,必須在窗口被刪除后,顯式地調(diào)用delete來(lái)刪除對(duì)象(一般在擁有者或父窗口的析構(gòu)函數(shù)中進(jìn)行).對(duì)于具有自動(dòng)清除功能的窗口對(duì)象,只需調(diào)用CWnd::DestroyWindow即可刪除窗口和窗口對(duì)象。注意,對(duì)于在堆中創(chuàng)建的窗口對(duì)象,不要在窗口還未關(guān)閉的情況下就用delete操作符來(lái)刪除窗口對(duì)象.