zoukankan      html  css  js  c++  java
  • 内存泄漏

    内存泄漏:

    程序申请了堆空间,可是“忘记”释放,导致该块区域在程序结束前无法被再次使用导致的。泄漏时间长了,就会导致用户空间内存不足,严重的导致死机。

    假设泄漏比較严重,非常easy察觉;可是有些泄漏非常缓慢,不easy察觉,可是软件会执行非常长时间后,会慢慢导致严重问题,并且当发现症状的时候,基本上已经是比較晚的时候了,想要识别泄漏,还是能够实现的,本篇文章来聊聊内存操作的原理。

    C++中申请内存使用的是new,C语言中使用的malloc(还有其它比方alloc原理类似),普通情况下new调用的是C语言使用的malloc。而因为C++/C能够在多个操作系统使用,所以能够想到malloc肯定是封装了不同操作系统提供的API,比方Windows上边调用的有这几个:VirtualAllocVirtualAllocExVirtualFreeVirtualProtectVirtualQueryVirtualLockVirtualUnLock,只是它们分配的都是内存页的整数倍的虚拟内存空间RAM,然而我们想要使用内存的时候,差点儿非常少须要整数倍,那么就须要封装它们改变灵活一些,API替我们封装了:HeapAllocGlobalAlloc,它们全都是内核级别,存在于kernel32.dll里边。

    非常easy看出,我们调用new,实际调用的是HeapAlloc,只是不须要在检測内存泄漏的时候走到这么底层,回归正传,还是回到new/malloc

    在申请内存的时候new调用malloc,释放delete调用free,那么怎么可以让每一次申请和每一次释放都可以记录,终于找到可能的哪怕1字节的泄漏呢?easy想到,在new调用malloc的时候,记录下申请空间的地址,在delete调用free的时候,将记录里边相应地址的数据删除,这样最后假设记录中还有剩余记录,则表示有泄漏。这样还不够,即使知道泄漏,但是不知道在哪有有什么用呢?显然在记录的时候,除了记录地址,还应记录申请的文件名称字、行数,这样最后有泄漏的时候,可以立马知道在哪里泄漏,原理就是这些,看下样例、解决的基础代码。


    #include
    
    #include
    
    using namespace std;
    
    #include
    
    
    
    
    void* operator new(size_t size, const char* file, int line)
    
    {
    
    void *p = (void*)malloc(size);
    
    printf("【%d】%s : (%d) 申请%d字节内存
    ", p,file,line,size);
    
    return p;
    
    }
    
    void* operator new[](size_t size, const char* file, int line)
    
    {
    
    return operator new(size, file, line);
    
    }
    
    #define new DEBUG_NEW
    
    #define DEBUG_NEW new(__FILE__, __LINE__)
    
    
    void operator delete(void* pointer)
    
    {
    
    printf("【%d】释放内存
    ",pointer);
    
    free(pointer);
    
    }
    
    
    void operator delete[](void* pointer)
    
    {
    
    operator delete(pointer);
    
    }
    
    
    void operator delete(void* pointer, const char* file, int line)
    
    {
    
    operator delete(pointer);
    
    }
    
    
    void operator delete[](void* pointer, const char* file, int line)
    
    {
    
    operator delete(pointer, file, line);
    
    }
    
    
    
    int main()
    
    {
    
    int *pos1 = new int[10];
    
    int *pos2 = new int(0x70000001);
    
    delete []pos1;
    
    delete pos2;;
    
    return 0;
    
    }
    这里边没有记录申请堆的文件名称字、行号,只是添加非常easy,用个HASH保存即可了,就不改动了。执行结果:


    easy发现申请和释放是成对的,释放内存地方本来也能够知道释放的大小的,只是有点忘记详细保存的方式了,C++编译器一般都会在new返回的仅仅针的前边几字节记录申请的长度等信息,能够在上边代码这么改动:


    就会出现崩溃,由于将编译器的数据改动了,它无法正常释放内存了。

    有个比較不错的开源内存泄漏检測项目:Visual Leak Detector,用它很方便,只须要将它的头文件、静态链接库加入到我们的项目中就可以,用上边的样例做样例:


    而当去掉一个delete 的时候,是这种:


    直接找到泄漏大小、地点。



    Visual Leak Detector下载地址:

    http://www.codeproject.com/script/articles/download.aspx?file=/KB/applications/visualleakdetector/vld-10.zip&rp=http://www.codeproject.com/Articles/9815/Visual-Leak-Detector-Enhanced-Memory-Leak-Detectio



    參考:

    http://www.ibm.com/developerworks/cn/linux/l-mleak2/

    http://blog.csdn.net/yapingxin/article/details/6751940

    http://bbs.csdn.net/topics/20329607

    http://blog.csdn.net/g5dsk/article/details/6077601

    http://babybandf.blog.163.com/blog/static/6199353201128101029894/

    http://www.codeproject.com/KB/applications/visualleakdetector/vld-10.zip

    http://blog.csdn.net/sunmenggmail/article/details/8316734
























  • 相关阅读:
    角色总结
    cookie
    基础php链接SQL数据库
    html
    PHP 每天的总结(1)
    php的特性
    [转载]CS0234: 命名空间“System.Data”中不存在类型或命名空间名称“OracleClien...
    [转载]数据库镜像中证书过期的解决方案
    华师大陈默老师的育儿讲
    [转载]如何使用VMware Workstation 8将物理机转换为虚拟机?
  • 原文地址:https://www.cnblogs.com/zfyouxi/p/4027072.html
Copyright © 2011-2022 走看看