記憶體泄漏及簡單檢測的一種方法

来源:http://www.cnblogs.com/shouce/archive/2016/04/29/5445020.html
-Advertisement-
Play Games

1.什麼是記憶體泄漏(Memory Leak)? 簡單地說就是申請了一塊記憶體空間,使用完畢後沒有釋放掉。它的一般表現方式是程式運行時間越長,占用記憶體越多,最終用盡全部記憶體,整個系統崩潰。由程式申請的一塊記憶體,且沒有任何一個指針指向它,那麼這塊記憶體就泄露了。 2.記憶體泄漏的危害性 從用戶使用程式的角度來 ...


1.什麼是記憶體泄漏(Memory Leak)? 

   簡單地說就是申請了一塊記憶體空間,使用完畢後沒有釋放掉。它的一般表現方式是程式運行時間越長,占用記憶體越多,最終用盡全部記憶體,整個系統崩潰。由程式申請的一塊記憶體,且沒有任何一個指針指向它,那麼這塊記憶體就泄露了。

2.記憶體泄漏的危害性

    從用戶使用程式的角度來看,記憶體泄漏本身不會產生什麼危害,作為一般的用戶,根本感覺不到記憶體泄漏的存在。真正有危害的是記憶體泄漏的堆積,這會最終消耗盡系統所有的記憶體。主要有以下幾種表現形式:

1)cpu資源耗盡:估計是機器沒有反應了,鍵盤,滑鼠,以及網路等等。這個在windows上經常看見,特別是中了毒。 2)進程id耗盡:沒法創建新的進程了,串口或者telnet都沒法創建了。 3)硬碟耗盡: 機器要死了,交換記憶體沒法用,日誌也沒法用了,死是很正常的。

在我們寫程式的時候,一般會使用malloc,realloc,new等函數從堆中分配到一塊記憶體,使用完後,程式必須負責相應的調用free或delete釋放該記憶體塊,否則,這塊記憶體就不能被再次使用,我們就說這塊記憶體泄漏了。如果要避免這個問題,還是要從代碼上入手,良好的編碼習慣和規範,是避免錯誤的不二法門。

3.如何檢測記憶體泄漏?

第一:良好的編碼習慣,儘量在涉及記憶體的程式段,檢測出記憶體泄露。當程式穩定之後,在來檢測記憶體泄露時,無疑增加了排除的困難和複雜度。使用了記憶體分配的函數,一旦使用完畢,要記得要使用其相應的函數釋放掉。

Heap memory:

malloc\realloc ------ free

new \new[] ---------- delete \delete[]

GlobalAlloc------------GlobalFree

要特別註意數組對象的記憶體泄漏

MyPointEX *pointArray =new MyPointEX [100];

其刪除形式為:

delete []pointArray ;


第二:將分配的記憶體的指針以鏈表的形式自行管理,使用完畢之後從鏈表中刪除,程式結束時可檢查改鏈表。

第三:Boost 中的smart pointer。

第四:一些常見的工具插件,如ccmalloc、Dmalloc、Leaky等等。

4.代碼示例

    我主要想結合代碼講講第二個方法,設計思想其實很簡單,用到STL中的list。可能有人要問,為什麼不用vector呢?
list和vector的區別如下:

vector為存儲的對象分配一塊連續的地址空間,因此對vector中的元素隨機訪問效率很高。在vecotor中插入或者刪除某個元素,需要將現有元素進行複製,移動。如果vector中存儲的對象很大,或者構造函數複雜,則在對現有元素進行拷貝時開銷較大,因為拷貝對象要調用拷貝構造函數。對於簡單的小對象,vector的效率優於list。vector在每次擴張容量的時候,將容量擴展2倍,這樣對於小對象來說,效率是很高的。

list中的對象是離散存儲的,隨機訪問某個元素需要遍歷list。在list中插入元素,尤其是在首尾插入元素,效率很高,
只需要改變元素的指針。

vector適用:對象數量變化少,簡單對象,隨機訪問元素頻繁

list適用:     對象數量變化大,對象複雜,插入和刪除頻繁

#include <iostream>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <list>

using namespace std;

const int nMaxSize = 26;
struct Node
{    
    int count;   
    Node *next[nMaxSize];
};
Node *root = NULL;
list <Node *> nodeMemory;

void TreeCreate()
{    
    root = (Node *)malloc(sizeof(Node)); 
    for (int i = 0; i < nMaxSize; ++i) 
    {       
        root->next[i] = NULL;   
    }
}

void TreeInsert(char *pStr)
{    
    int i, j, len = strlen(pStr);
    Node *p = root;    
    Node *q = NULL; 
    for (i = 0; i < len; ++i) 
    {        
        int id = pStr[i] - 'a';    
        if (p->next[id] == NULL)
        {          
            q = (Node *)malloc(sizeof(Node)); 
            if(q != NULL)
                nodeMemory.push_back(q);
            q->count = 0;  
            for (j = 0; j < nMaxSize; ++j)
            {            
                q->next[j] = NULL; 
            }         
            p->next[id] = q; 
        }      
        p->next[id]->count++;  
        p = p->next[id];   
    }
}

int TreeQuery(char *pStr)
{   
    int i, len = strlen(pStr); 
    int id = 0;   
    Node *p = root;  
    for (i = 0; i < len; ++i)  
    {        
        id = pStr[i] - 'a';  
        p = p->next[id];      
        if (p == NULL) return 0;  
    }       
    return p->count;
}

void TreeDelete(Node *p)
{   
    if (p == NULL) 
        return ;  
    for (int i = 0; i < nMaxSize; ++i) 
    {      
        TreeDelete(p->next[i]);   
    }    
    nodeMemory.remove(p);
    free(p);
    p = NULL;
}

int main(int argc, char **argv)
{   
    char szBuffer[16]; 
    int res = 0;      
    TreeCreate(); 
    int n = 3;
    while (n)   
    {       
        gets(szBuffer); 
        if (strlen(szBuffer) == 0)
            break;     
        TreeInsert(szBuffer); 
        n--;
    }  
/*    scanf("%s", szBuffer);       
    res = TreeQuery(szBuffer); 
    printf("%d\n", res);  */ 
    for(list<Node *>::iterator it = nodeMemory.begin();it != nodeMemory.end();it++)
    {
        cout<<*it<<endl;
    }
    cout<<nodeMemory.size()<<endl;
    TreeDelete(root);    
    if(nodeMemory.empty())
        cout<<"has delete the tire tree"<<endl;
    cout<<nodeMemory.size()<<endl; 
    system("pause");
    return 0;
}

  

代碼很簡單,list存儲的是指向記憶體空間的指針,每次malloc之後會把這個分配的記憶體指針push到list中,而當free之後,list的就會刪除相應的內容,如果全釋放掉了,則list變為空,從而可以判斷使用後的記憶體是否全部釋放掉了?


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 最近遇到了一個非常邪門的故障,重新安裝了Windows2008伺服器後 Mp4無法正常播放; 整個互聯網中關於設置MP4的方法基本都是教你如何在IIS中設置MIME 本文討論的不是如何教你設置MIME,如果當你配置完MIME後仍然無法播放MP4,那麼您就要檢查一下是否安裝了《安全狗》軟體。 當前時間 ...
  • Linq的delegate表達式,Insus.NET覺得它封裝得好,讓開發時簡化了很多代碼,而且容易閱讀與檢索。 比如,我們需要計算優惠給客戶金額,打85%折,可以這樣寫: using System; using System.Collections.Generic; using System.Li ...
  • 在ASP.NET MVC 小牛之路系列中,前面用了一篇文章提了一下C#的一些知識點。照此,ASP.NET MVC 大牛之路系列也先給大家普及一下C#.NET中的高級知識點。每個知識點不太會過於詳細,但足矣。要深入研究還需要去查看更多的專業資料。 要成為大牛,必然要有扎實的基本功,不然時間再長項目再多 ...
  • 因為公司有多個伺服器,要檢查磁碟的使用情況確定程式放哪個伺服器和清理垃圾,所以寫個小程式幫忙檢查。 效果圖: 後臺代碼: ...
  • 近乎的Ajax控制項介紹,代碼下載:http://www.jinhusns.com/Products/Download?type=whp AjaxForm 概述 功能說明 基於 ajaxForm 插件進行構建,可以使表單通過ajax的方式進行提交,並擴展了以下功能: 1.通過Html.BeginAja ...
  • 昨天在手機端測試自己的項目遇到如下情況。 1.在手機上(蘋果qq瀏覽器),登陸時存在session中的圖片驗證碼結果,一直獲取不到,考慮是cookies的問題。但是其他網站有貌似可以正常使用cookies,這是第一點矛盾。 2.在平板上(安卓qq瀏覽器),一直可以正常使用session,證明cook ...
  • ()help('What your wannt object') 看對像怎麼用法,按key q退出()在Python中有4種類型的數——整數、長整數、浮點數和複數。 * 2是一個整數的例子。 * 長整數不過是大一些的整數。 * 3.23和52.3E-4是浮點數的例子。E標記表示10的冪。在這裡,52 ...
  • 首先關於json_encode的gbk中文亂碼,創建JSON函數,這一段來自網上某一位大俠 /************************************************************** * * 使用特定function對數組中所有元素做處理 * @param str ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...