扩展阅读
  • 应用服务器和Web服务器的区别?
  • 如何将日志从一台UNIX服务器传到另外一台UNIX服务器?
  • 我断开socket服务器日志之后,服务器程序就挂死啦
  • 怎么用javabean调用服务器的dll或访问服务器的com口
  • linux服务器做WEB服务器,页面一片空白是什么原因?
  • 知道最便宜的应用服务器和数据库服务器吗?
  • linux下web服务器与数据库服务器的分离问题(在线等!非常急!)
  • 对于邮件服务器更换中,2台邮件服务器同时运行,有何策略?公司实例
  • 请问JDBC服务器与WEB服务器是否一个概念?
  • 品牌服务器与杂牌服务器之争议---(路过也有分)
  • qmail启用smtp认证后,为什么别的邮件服务器不能发信到我的服务器。。。
  • 一个最简单的问题:www服务器与web服务器是否一个概念?
  • 98用telnet到linux服务器上,怎样把服务器上的文件拷到本地硬盘上
  • putty ssh登录如何将文件上传至服务器,如何从服务器下载数据到本地?
  • 打开SecureCRT后,如何用命令连到某一台linux服务器,并用命令下载服务器上的某个文件
  • 请问配置了ftp服务器、邮件服务器后系统默认开放哪些端口?
  • 怎么样实现由Linux服务器映射访问发布Websphere的win服务器!
  • CentOS6.4 作为邮件服务器和Web服务器应该怎么分区?
  • LINUX下如何将拔号服务器配置为通过RADIUS服务器验证!最好详细一点!
  • 求助! telnet到远程服务器后,如何访问服务器端字符设备
  •  
    当前位置:  编程语言>c/c++

    c/c++服务器程序内存泄露问题分析及解决

     
        发布时间:2013-7-23  


        本文导语:  对于一个c/c++程序员来说,内存泄漏是一个常见的也是令人头疼的问题。已经有许多技术被研究出来以应对这个问题,比如 Smart Pointer,Garbage Collection等。Smart Pointer技术比较成熟,STL中已经包含支持Smart Pointer的class,...

       对于一个c/c++程序员来说,内存泄漏是一个常见的也是令人头疼的问题。已经有许多技术被研究出来以应对这个问题,比如 smart pointergarbage collection等。smart pointer技术比较成熟,stl中已经包含支持smart pointer的class,但是它的使用似乎并不广泛,而且它也不能解决所有的问题;garbage collection技术在java中已经比较成熟,但是在c/c++领域的发展并不顺畅,虽然很早就有人思考在c++中也加入gc的支持。现实世界就是这样的,作为一个c/c++程序员,内存泄漏是你心中永远的痛。不过好在现在有许多工具能够帮助我们验证内存泄漏的存在,找出发生问题的代码。

    1.内存泄漏的定义

       一般我们常说的内存泄漏是指堆内存的泄漏。堆内存是指程序从堆中分配的,大小任意的(内存块的大小可以在程序运行期决定),使用完后必须显示释放的内存。应用程序一般使用mallocreallocWindows iis7站长之家函数从堆中分配到一块内存,使用完后,程序必须负责相应的调用freedelete释放该 内存块,否则,这块内存就不能被再次使用,我们就说这块内存泄漏了。以下这段小程序演示了堆内存发生泄漏的情形:

    void MyFunction(int nSize)
    {
     char* p= new char[nSize];
     if( !GetStringFrom( p, nSize ) ){
      MessageBox(“Error”);
      return;
     }
     …//using the string pointed by p;
     delete p;
    }

    当函数GetStringFrom()返回零的时候,指针p指向内存就不会被释放。这是一种常见的发生内存泄漏的情形。程序在入口处分配内存,在出口处释放内存,但是c函数可以在任何地方退出,所以一旦有某个出口处没有释放应该释放的内存,就会发生内存泄漏。

      广义的说,内存泄漏不仅仅包含内存的泄漏,还包含系统资源的泄漏(resource leak),比如核心态HANDLEGDI ObjectSOCKETInterface等,从根本上说这些由操作系统分配的对象也消耗内存,如果这些对象发生泄漏最终也会导致内存的泄漏。而且,某些对象消耗的是核心态内存,这些对象严重泄漏时会导致整个操作系统不稳定。所以相比之下,系统资源的泄漏比堆内存的泄漏更为严重。

      GDI Object的泄漏是一种常见的资源泄漏:

    void CMyView::OnPaint( CDC* pDC )
    {
     CBitmap bmp;
     CBitmap* pOldBmp;
     bmp.LoadBitmap(IDB_MYBMP);
     pOldBmp = pDC->SelectObject( &bmp );
     …
     if( Something() ){
      return;
     }
     pDC->SelectObject( pOldBmp );
     return;
    }

    当函数Something()返回非零的时候,程序在退出没有把pOldBmp选回pDC中,这会导致pOldBmp指向的HBITMAP对象发生泄 漏。这个程序如果长时间的运行,可能会导致整个系统花屏。这种问题在Win9x下比较容易暴露出来,因为Win9x的GDI堆比Win2k或NT的要小很 多。

      内存泄漏的发生方式:

      以发生的方式来分类,内存泄漏可以分为4

      1) 常发性内存泄漏。发生内存泄漏的代码会被多次执行到,每次被执行的时候都会导致一块内存泄漏。比如例二,如果Something()函数一直返回True,那么pOldBmp指向的HBITMAP对象总是发生泄漏。

       2) 偶发性内存泄漏。发生内存泄漏的代码只有在某些特定环境或操作过程下才会发生。比如例二,如果Something()函数只有在特定环境下才返回 True,那么pOldBmp指向的HBITMAP对象并不总是发生泄漏。常发性和偶发性是相对的。对于特定的环境,偶发性的也许就变成了常发性的。所以 测试环境和测试方法对检测内存泄漏至关重要。

      3) 一次性内存泄漏。发生内存泄漏的代码只会被执行一次,或者由于算法上的缺陷,导致总会有一块仅且一块内存发生泄漏。比如,在类的构造函数中分配内存,在析 构函数中却没有释放该内存,但是因为这个类是一个Singleton,所以内存泄漏只会发生一次。另一个例子:

    char* g_lpszFileName = NULL;
    void SetFileName( const char* lpcszFileName )
    {
     if( g_lpszFileName ){
      free( g_lpszFileName );
     }
     g_lpszFileName = strdup( lpcszFileName );
    } 

    如果程序在结束的时候没有释放g_lpszFileName指向的字符串,那么,即使多次调用SetFileName(),总会有一块内存,而且仅有一块内存发生泄漏。

       4)隐式内存泄漏。程序在运行过程中不停的分配内存,但是直到结束的时候才释放内存。严格的说这里并没有发生内存泄漏,因为最终程序释放了所有申请的内存。但 是对于一个服务器程序,需要运行几天,几周甚至几个月,不及时释放内存也可能导致最终耗尽系统的所有内存。所以,我们称这类内存泄漏为隐式内存泄漏。举一 个例子:

    class Connection
    {
     public:
      Connection( SOCKET s);
      ~Connection();
      …
     private:
      SOCKET _socket;
      …
    };
    class ConnectionManager
    {
     public:
      ConnectionManager(){}
      ~ConnectionManager(){
       list::iterator it;
       for( it = _connlist.begin(); it != _connlist.end(); ++it ){
        delete (*it);
       }
       _connlist.clear();
      }
      void OnClientConnected( SOCKET s ){
       Connection* p = new Connection(s);
       _connlist.push_back(p);
      }
      void OnClientDisconnected( Connection* pconn ){
       _connlist.remove( pconn );
       delete pconn;
      }
     private:
      list _connlist;
    };

    假设在ClientServer端断开后,Server并没有呼叫OnClientDisconnected()函数,那么代表那次连接Connection对象就不会被及时的删除(在Server程序退出的时候,所有Connection对象会在ConnectionManager的析 构函数里被删除)。当不断的有连接建立、断开时隐式内存泄漏就发生了。

      从用户使用程序的角度来看,内存泄漏本身不会产生什么危害,作为一般的用户,根本感觉不到内存泄漏的存在。真正有危害的是内存泄漏的堆积,这会最终消耗尽系统所有的内存。从这个角度来说,一次性内存泄漏并没有什么危 害,因为它不会堆积,而隐式内存泄漏危害性则非常大,因为较之于常发性和偶发性内存泄漏它更难被检测到。

    2.检测内存泄漏

      检测内存泄漏的关键是要能截获住对分配内存和释放内存的函数的调用。截获住这两个函数,我们就能跟踪每一 块内存的生命周期,比如,每当成功的分配一块内存后,就把它的指针加入一个全局list中;每当释放一块内存,再把它的指针从list中删除。这样,当 程序结束的时候,list中剩余的指针就是指向那些没有被释放的内存。这里只是简单的描述了检测内存泄漏的基本原理,详细的算法可以参见Steve Maguire的<<Writing Solid Code>>。

      如果要检测堆内存的泄漏,那么需要截获住 malloc/realloc/free和new/delete就可以了(其实new/delete最终也是用malloc/free的,所以只要截获前 面一组即可)。对于其他的泄漏,可以采用类似的方法,截获住相应的分配和释放函数。比如,要检测BSTR的泄漏,就需要截获 SysAllocString/SysFreeString;要检测HMENU的泄漏,就需要截获CreateMenu/ DestroyMenu。(有的资源的分配函数有多个,释放函数只有一个,比如,SysAllocStringLen也可以用来分配BSTR,这时就需要 截获多个分配函数)

      在Windows平台下,检测内存泄漏的工具常用的一般有三种,MS C-Runtime Library内建的检测功能;外挂式的检测工具,诸如,Purify,BoundsChecker等;利用Windows NT自带的Performance Monitor。这三种工具各有优缺点,MS C-Runtime Library虽然功能上较之外挂式的工具要弱,但是它是免费的;Performance Monitor虽然无法标示出发生问题的代码,但是它能检测出隐式的内存泄漏的存在,这是其他两类工具无能为力的地方。  

      内存泄漏是个大而复杂的问题,即使是java和.net这样有 gabarge collection机制的环境,也存在着泄漏的可能,比如隐式内存泄漏。

    3.避免内存泄露

      写服务器程序,最怕的就是内存泄露。因为程序经常运行好几个月不停,一点点内存泄露都会导致悲剧的发生。常规来说,首要是避免内存泄露,其次是检查内存泄露。

    1)不用new

    c++程序,尽量多用stl,避免用new。我自己写的代码,除了在main函数里面有new外,其他地方不会再有任何new出现。这样就把内存管理交给stl去做。

    或许你会说,不用new怎么可能啊?

    很简单,char数组std::string代替,其他对象直接拷贝。除非你的对象很大很大,否则,一点点拷贝耗时,完全可以忽略不计。

    2)每个重要结构都提供Info函数

    给你的每个重要结构都加上一个Info函数,info函数返回一个string,描述当前结构的状态,如map的大小,内存占用的大小。在最顶层,不定时的输出(或者根据命令输出)各个对象的info结果。这样可以避免隐形的内存泄露,即不是内存泄露,但某个对象保持的大量对象的引用,导致对象无法被删除;

    即某个对象内部的map,不断的添加数据,也在不断的删除数据,但在某些特殊情况下,它不会删除。

    3)stl内存泄露的问题

    stl几乎没有内存泄露,但它有一个内存cache,这个cache对小对象的分配很友好。stl的一个麻烦是,它几乎不会释放这些空间,这样的一个结果是,你看到自己的程序内存占用不断的上涨。其实,理论上是不用害怕的,因为它涨到一定范围(如,机器只有几百兆可用空间了),就不会涨了。可以通过在运行程序前,export glibcxx_force_new=1,来让stl不要进行cache。注意,这个仅仅在gcc(g++) 3.3以后有效。

    4)valgrind内存检测工具

     直接下载编译(注意,必须在configure的当前目录下执行configure,不能另外选一个目录),安装。执行:valgrind --num-callers=20 --leak-check=full --leak-resolution=high --show-reachable=yes --log-file=val.log xxx &,等过了几天后,把它kill,然后慢慢的看val.log文件

    当你采用了前面3个策略后,valgrind几乎没有啥效果,反正我从来没有从它这里获得任何有用的信息过。主要是因为前面几步保证了没有显式的内存泄露,所以,valgrind也就找不出来啥内存泄露了。

    5)valgrind的工具massif

    massif比valgrind好的地方在于,它会告诉你当前内存的分布情况。你可以看到占用了几百兆的程序到底是那些地方占用了内存。执行:valgrind --tool=massif xxx。一般通过这个都可以看到明显的内存泄露。这个工具很好,俺用它发现了一个十分异常的情况,这个情况是由vector的reserve导致的。本来应该reserve返回数据的个数,结果reserve了命中结果的个数。导致偶尔会出现内存占用过500M的情况,但因为没有内存泄露,所以其他几个工具都找不出来,就只有massif提供的堆栈快照可以发现这个问题。


    • 本站(WWW.)旨在分享和传播互联网科技相关的资讯和技术,将尽最大努力为读者提供更好的信息聚合和浏览方式。
      本站(WWW.)站内文章除注明原创外,均为转载,整理或搜集自网络.欢迎任何形式的转载,转载请注明出处.
      转载请注明:文章转载自:[169IT-IT技术资讯]
      本文标题:c/c++服务器程序内存泄露问题分析及解决
    相关文章推荐:
  • 重装服务器后IIS网站错误(应用程序中的服务器错误)
  • A服务器的shell脚本调用B服务器的shell脚本,怎样使B服务器shell执行返回值返回给A服务器的脚本
  • nginx 服务器介绍
  • 请问在红旗Linux多功能服务器版上(不是数据库服务器版)能否正常安装使用Oracle?红旗Linux数据库服务器版要比红旗Linux多功能服务器版贵
  • 代理服务器原理,功能及作用详细介绍
  • 请教想用linux做前置服务器,后面放一些web服务器,数据库服务器啥的
  • Apache HTTP服务器2.4中的新功能概览
  • 校园网,www服务器,邮件服务器,数据库服务器等各用什么平台(系统)性能最佳?
  • php中操作memcache的类及成员列表及php下如何连接memched服务器
  • 双linux服务器如何能在一台服务器崩溃的情况下在另一台服务器上启动相应服务
  • nginx服务器基本特性及使用技巧
  • 如何有效判断服务器已经连接不上(服务器关闭,服务器非法退出等原因)
  • nginx服务器下通过fastcgi支持php5详细安装配置步骤
  • 在服务器端怎样检测客户端与服务器端的某个会话在一段时间里面没有通信,然后在服务器端终止该会话??
  • Mysql服务器登陆,启动,停止等基本操作命令介绍(Linux/Centos环境)
  • 如果服务器不是本机(程序在一服务器,数据库在一服务器,局域网),如何连接sql ??
  • 服务器存储快照和数据库快照详解
  • 采用sendmail搭建邮件服务器必须在邮件服务器上构建DNS服务吗?
  • web服务器和应用服务器的区别?
  • 如何去理解web服务器与应用服务器
  • 现在有50台服务器,他们的IP是192.168.1.1~192.168.1.100,如何将服务器192.168.1.1的/etc/rc.local快速复制到另外49台服务器上?


  • 站内导航:


    特别声明:169IT网站部分信息来自互联网,如果侵犯您的权利,请及时告知,本站将立即删除!

    ©2012-2021,,E-mail:www_#163.com(请将#改为@)

    浙ICP备11055608号-3