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