• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            MyMSDN

            MyMSDN記錄開發新知道

            關于內存泄露的問題(解決+剖析)

            首先先闡明這篇隨筆的意圖,只在告訴讀者,內存泄露的神不知鬼不覺,希望能引起大家的注意。
            一段代碼的意思如何正確表達,才能不造成內存泄露呢?很多朋友經常泄露了內存但卻查找不到原因。當然在CLI/C++中利用托管對象堆上的垃圾收集器是可以更好地避免這一點。但是在更早的版本中,程序員有必要去手動刪除這些相關資源。否則將在程序關閉的時候出現一些錯誤。
            MFC
            現在我們去重載一個虛函數virtualvoidDeleteContents();用來在銷毀文檔數據前調用框架刪除一些文檔類的數據,(MSDN:Called by the framework to delete the document's data without destroying the CDocument object itself.)
            先批評一段代碼:

            1 void ?CGraphicDoc::DeleteContents()?
            2 {
            3 ???? for ( int ?i = 0 ;i < m_obArray.GetSize();i ++ )
            4 ???? {
            5 ????????……
            6 ????}

            7 ????CDocument::DeleteContents();
            8 }
            評價:這段代碼看似簡練,但是卻很浪費資源,在第3行的for循環中,i<m_obArray.GetSize();當每次進行判斷的時候將再次調用GetSize(),如果這個數據的量是個天文數字,那么這樣的調用無疑是一種災難。

            優化:
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????for(int?i=0;i<nCount;i++)
            ?6????{
            ?7????????……
            ?8????}

            ?9????CDocument::DeleteContents();
            10}

            填寫for循環內的語句。這里的任務:刪除之前利用CObArray : m_obArray對象保存的一個指針所指向的對象,以及指針本身。因此(以下提供幾種常見的錯誤代碼)
            代碼A:
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????for(int?i=0;i<nCount;i++)
            ?6????{
            ?7????????delete?m_obArray.GetAt(i);???//刪除對象指針所指向的對象
            ?8????????m_obArray.RemoveAt(i);???//刪除對應的指針本身
            ?9????}

            10????CDocument::DeleteContents();
            11}
            代碼B:
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????for(int?i=0;i<nCount;i++)
            ?6????{
            ?7????????delete?m_obArray.GetAt(i);
            ?8????????m_obArray.RemoveAt(0);
            ?9????}

            10????CDocument::DeleteContents();
            11}
            代碼A看起來似乎很符合常規思維,因此也很容易迷惑人。但是在程序運行的時候出現了錯誤。但是在MSDN中查找CObArray Class Members,查看RemoveAt,其中remarks中有這樣一句:In the process, it shifts down all the elements above the removed element(s). It decrements the upper bound of the array but does not free memory. 它的意思就是假設你一共有3個數據 p[0]=a、p[1]=b、p[2]=c,當你刪除第0個數據之后,數據將整體向前移動,變為p[0]=b、p[1]=c。這樣當你i=nCount-1的時候就根本沒有這樣的數讓你刪除,因為假設有那個時刻的話,數據的元素只有1個,而他的編號是0而不是nCount.因此出現了無法刪除的現象。因此也就隱含了問題了。
            因此有人突發奇想:如果我每次只刪除第0個數據的話,那么是否就可以了呢?于是代碼B誕生了。可是問題終究沒能得到解決。因為假設有一組數據一共3個。刪除了編號0的元素(delete語句),移除了該元素的指針,此時i=1,進入刪除,又到了delete語句,這時候刪除元素i=1這樣的語句,這時實際上是刪除了先前元素中的第二個元素,而不是第一個。而0與2中間的第1個元素則未被刪除。又出現了隱含問題。其實只要將兩個都改為0,每次都刪除第一個就可以了。
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????for(int?i=0;i<nCount;i++)
            ?6????{
            ?7????????delete?m_obArray.GetAt(0);
            ?8????????m_obArray.RemoveAt(0);
            ?9????}

            10????CDocument::DeleteContents();
            11}
            另外可以將程序倒寫過來,避免RemoveAt對其進行重新整合隊列做產生的不可預料的麻煩。
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????while(nCount--)
            ?6????{
            ?7????????delete?m_obArray.GetAt(nCount);
            ?8????????m_obArray.RemoveAt(nCount);
            ?9????}

            10????CDocument::DeleteContents();
            11}

            12
            或者仔細察看MSDN中還有一個函數叫RemoveAll()它是用來刪除整個CObArray集合對象的。
            ?1void?CGraphicDoc::DeleteContents()?
            ?2{
            ?3????int?nCount;
            ?4????nCount=m_obArray.GetSize();
            ?5????for(int?i=0;i<nCount;i++)
            ?6????{
            ?7????????delete?m_obArray.GetAt(i);
            ?8????}

            ?9????m_obArray.RemoveAll();
            10????CDocument::DeleteContents();
            11}

            posted on 2006-08-16 03:44 volnet 閱讀(4181) 評論(5)  編輯 收藏 引用

            評論

            # re: 關于內存泄露的問題(解決+剖析) 2006-08-16 08:54 szwolf

            寫得非常好: )
            對MFC內部的細節了解得很詳細,C++內存管理思路也良清淅。  回復  更多評論   

            # re: 關于內存泄露的問題(解決+剖析) 2006-08-16 09:15 小明

            另外一種我經常用的寫法
            for(int i=0,nCount = m_obArray.GetSize();i<nCount;++i)
            {
            delete m_obArray.GetAt(i);
            }

            另外我不建議在循環中RemoveAt,效率太差。  回復  更多評論   

            # re: 關于內存泄露的問題(解決+剖析) 2006-08-16 09:16 mzty

            en ,不錯的啊  回復  更多評論   

            # re: 關于內存泄露的問題(解決+剖析) 2006-08-16 10:08 volnet

            呵呵,謝謝大家,你們的評論讓我更有信心多寫好文章
            發文章最怕沒人看了。和存檔案一樣,還不如存自家硬盤呢。。。。
            所以看文章,+評論,大家一起互動~  回復  更多評論   

            # re: 關于內存泄露的問題(解決+剖析) 2009-07-18 14:31 cs

            寫技術博客很辛苦的哦,謝謝你了  回復  更多評論   

            特殊功能
             
            av无码久久久久久不卡网站| 久久人人爽人人爽人人片AV高清| 色综合久久久久久久久五月| 伊人久久大香线焦AV综合影院| 综合网日日天干夜夜久久| 欧美一区二区三区久久综合| 一级做a爰片久久毛片人呢| 久久久久久久综合日本| 亚洲欧美日韩中文久久| 久久国产一片免费观看| 日韩人妻无码精品久久久不卡 | 久久久久国产一区二区三区| 久久综合久久综合亚洲| 久久免费高清视频| 色婷婷久久综合中文久久蜜桃av| 久久精品中文字幕有码| 国产成人无码久久久精品一| 免费久久人人爽人人爽av| 青青青国产成人久久111网站| 天天综合久久一二三区| AV狠狠色丁香婷婷综合久久 | 欧美精品国产综合久久| 久久发布国产伦子伦精品| 久久婷婷色综合一区二区| aaa级精品久久久国产片| 亚洲午夜久久久久久久久久| 久久本道久久综合伊人| 青青国产成人久久91网| 久久免费视频观看| 国产午夜精品理论片久久影视| 久久人人爽人人爽人人片AV高清| 久久一区二区三区99| 久久精品国产黑森林| 成人午夜精品久久久久久久小说| 丰满少妇人妻久久久久久| 亚洲∧v久久久无码精品| 久久精品免费全国观看国产| 久久91精品国产91| 中文字幕久久久久人妻| 无码专区久久综合久中文字幕| 久久久久青草线蕉综合超碰|