TCMalloc 安装和使用

  • Author:Echo Chen(陈斌)

  • Email:chenb19870707@gmail.com

  • Blog:Blog.csdn.net/chen19870707

  • Date:October 20th, 2014

    前面三篇译文《TCMalloc:线程缓冲的Malloc》、《使用TCMalloc的堆栈检查》、《使用TCMalloc进行堆栈分析》介绍了TCMalloc的基本原理,以及堆栈分析和检查工具,TCMalloc长处非常多。比glibc
    2.3的malloc快、自带的堆栈工具能够轻松找出内存瓶颈和内存泄漏。给server开发指明了一条新的道路。

    一、下载

    google-perftools:http://code.google.com/p/google-perftools/gperftools-2.1.tar.gz

    libunwind:http://download.savannah.gnu.org/releases/libunwind/libunwind-1.1.tar.gz

    二、libunwind安装

    64位操作系统请先安装 libunwind库,32位操作系统不要安装。libunwind库为基于64位CPU和操作系统的程序提供了主要的堆栈辗转开解功能。当中包含用于输出堆栈跟踪的API、用于以编程方式辗转开解堆栈的API以及支持C++异常处理机制的API。

       1: #tar zxvf libunwind-1.1.tar.gz
       2: #cd libunwind-1.1
       3: #./configure
       4: #make
       5: #make install

    三、安装google-perftools:

       1: #tar zxvf tar zxvf gperftools-2.1.tar.gz 
       2: #cd gperftools-2.1
       3: #./configure
       4: #make
       5: #make install

    四、TCMalloc库载入到Linux系统中:

       1: echo "/usr/local/lib" > /etc/ld.so.conf.d/usr_local_lib.conf
       2: /sbin/ldconfig

    五、使用

    在Makefile中 加入TCMalloc链接,注意:这里为了进行内存泄漏分析,一定要将TCMalloc链接在最后,官方文档里说:堆栈检查器可能误解列在它后面的链接库的一些内存。

       1: # funshion wuhan game studio
       2: # Created by zeng jun fei in 2013-08-08
       3:  
       4: CXX = g++
       5: # debug
       6: CXXFLAGS =  -g -I../BaseCode -I../../CommonSrc -Wall -D_POSIX_MT_ -O0
       7: CXXLFLAGS =  -g -Wall -L../bin -lBaseCode -lpthread  -lprotobuf -rdynamic -ltcmalloc
       8:  
       9: # release
      10: # CXXFLAGS =  -O3 -g -I../NetworkEngine -Wall
      11: # CXXLFLAGS =  -O3 -g -Wall -L../NetworkEngine -lnetwork
      12:  
      13: LIB_NETWORK = ../bin/libBaseCode.a
      14:  
      15: OBJS = $(patsubst %.cpp,%.o,$(wildcard *.cpp))
      16: SRCS = $(OBJS:%.o=%.cpp)
      17: DEPS = $(OBJS:%.o=%.d)
      18:  
      19: ALL_TARGETS = ../bin/GateServer
      20:  
      21: all: $(ALL_TARGETS)
      22:  
      23: -include $(DEPS)
      24: $(DEPS): %.d: %.cpp
      25:     @$(CXX) -MM $(CXXFLAGS) $< > $@.$$$$; sed 's,\($*\)\.o[ :]*,\1.o $@ : ,g' < $@.$$$$ > $@; rm -f $@.$$$$
      26:  
      27: $(OBJS): %.o: %.cpp
      28:     $(CXX) -c $(CXXFLAGS) $< -o $@
      29:  
      30: $(ALL_TARGETS): $(OBJS) $(LIB_NETWORK)
      31:     $(CXX) $(OBJS) -o $@ $(CXXLFLAGS)
      32:  
      33: clean:
      34:     @rm -rf $(OBJS) $(ALL_TARGETS) *.d
      35:  

    六、堆栈检查和分析

    首先,设置pperf的环境变量:export PPROF_PATH=/usr/local/bin/pprof

    測试代码:

       1: #include <iostream>
       2: using namespace std;
       3:  
       4: int main()
       5: {
       6:         int *p = new int();
       7:         return 0;
       8: }

    编译:g++ main.cpp -o main -ltcmalloc -g -O0

    内存泄漏检查:   env HEAPCHECK=normal ./main

    结果:

    WARNING: Perftools heap leak checker is active -- Performance may suffer

    Have memory regions w/o callers: might report false leaks

    Leak check _main_ detected leaks of 4 bytes in 1 objects

    The 1 largest leaks:

    Using local file ./main.

    Leak of 4 bytes in 1 objects allocated from:


        @ 4007a6 main

        @ 7f1734263d1d __libc_start_main

        @ 4006d9 _start

    If the preceding stack traces are not enough to find the leaks, try running THIS shell command:

    pprof ./main "/tmp/main.54616._main_-end.heap" --inuse_objects --lines --heapcheck  --edgefraction=1e-10 --nodefraction=1e-10 --gv

    If you are still puzzled about why the leaks are there, try rerunning this program with HEAP_CHECK_TEST_POINTER_ALIGNMENT=1 and/or with HEAP_CHECK_MAX_POINTER_OFFSET=-1


    If the leak report occurs in a small fraction of runs, try running with TCMALLOC_MAX_FREE_QUEUE_SIZE of few hundred MB or with TCMALLOC_RECLAIM_MEMORY=false, it might help find leaks more repeatably


    Exiting with error code (instead of crashing) because of whole-program memory leaks

    上面的报告显示有4个字节的内存泄漏,并提示使用pprof进一步跟踪泄漏来源的方法。

           包含normal在内总共同拥有4种泄漏检查方式:minimal,忽略进入main函数之前的初始化过程;normal。报告全部的无法再引用的内存对象。strick,在normal的基础上添加一些额外的检查;draconian。在程序退出的时候存在未释放的内存的情况下报错。

    依据《使用TCMalloc的堆栈检查》,除了前面使用env命令行的全局内存泄漏检查方式外,还能够作对代码段的更加细粒度的泄漏检查。

    这里须要先在源码中包括头文件google/heap-checker.h。

    实例代码例如以下:

       1: #include <cstdio>
       2: #include <cstdlib>
       3: #include <cassert>
       4: #include <google/heap-checker.h>
       5: int* fun(int n)
       6: {
       7:     int *p2;
       8:     HeapLeakChecker heap_checker("fun");
       9:     {
      10:         new int[n];
      11:         p2=new int[n];
      12:         //delete [] p1;
      13:     }
      14:     assert(!heap_checker.NoLeaks());
      15:     return p2;    
      16: }
      17: int main(int argc,char* argv[])
      18: {
      19:     int n;
      20:     scanf("%d",&n);
      21:     int *p=fun(n);
      22:     delete [] p;
      23:     return 0;
      24: } 

    此外,还能够忽略某些已知的内存泄漏:

       1: #include 
       2: ...
       3: void *mark = HeapLeakChecker::GetDisableChecksStart();
       4: <leaky code>
       5: HeapLeakChecker::DisableChecksToHereFrom(mark);

    感觉跟valgrind效果差点儿相同,可是valgrind还能查出内存越界。更加优秀。

    七、总结

    本来研究TCMalloc是为了优化游戏server,解决游戏server后期玩家流失后。占用大量内存的浪费。结果发现因为我们游戏server为了防止内存碎片和频繁调用new和delete带来的性能损耗。使用了大量的内存池对象(如装备池、技能池、玩家池),这些池都不会调用delete还给系统,所以即使使用了TCMalloc也不会有内存释放,如今也明确了server维护的意义,当然这和server框架设计非常有关系,假设没有这些缓冲池,直接调用new和delete,TCMalloc会是一个非常好的选择。

    -

    Echo Chen:Blog.csdn.net/chen19870707

    -

  • 版权声明:本文博主原创文章。博客,未经同意不得转载。

    TCMalloc 安装与使用的更多相关文章

    1. TCMalloc 安装和使用

      前面三篇译文<TCMalloc:线程缓冲的Malloc>.<使用TCMalloc的堆栈检查>.<使用TCMalloc进行堆栈分析>介绍了TCMalloc的基本原理, ...

    2. tcmalloc安装

      环境是centos 6 (64位) yum list libunwind-devel  (epel 源) wget https://gperftools.googlecode.com/files/gp ...

    3. Gperftools中tcmalloc的简介和使用(转)

      TcMalloc(Thread-CachingMalloc)是google-perftools工具中的一个内存管理库,与标准的glibc库中malloc相比,TcMalloc在内存分配的效率和速度上要 ...

    4. 源码部署pxc集群

      想了想还是研究一下怎么源码安装pxc吧,毕竟很多组件都是源码安装的. 环境: yum install -y boost-devel libodb-boost-devel check-devel ope ...

    5. [转]JVM内存模型

      最近排查一个线上java服务常驻内存异常高的问题,大概现象是:java堆Xmx配置了8G,但运行一段时间后常驻内存RES从5G逐渐增长到13G #补图#,导致机器开始swap从而服务整体变慢.由于Xm ...

    6. CentOS运维常用管理操作命令

      自己整理的整理Linux常用运维和linux常用管理操作命令,当然不是非常详细和丰富,但是也基本上够用了吧.欢迎留言补充更多的Linux常用运维和linux常用管理操作命令.不断完善中.... 备份m ...

    7. mysql安装tcmalloc

      TCMalloc(Thread-Caching Malloc)是google-perftools工具中的一个,与标准的glibc库的malloc相比,TCMalloc在内存的分配上效率和速度要高得多, ...

    8. redis(二)redis+TCMALLOC高性能的缓存服务器的安装配置

      安装  1准备编译环境    yum -y install gcc gcc+ gcc-c++ openssl openssl-devel pcre pcre-devel  2 下载源码包(由于goog ...

    9. TcMalloc的介绍以及Windows下安装使用

      本文由博主(SunboyL)原创,转载请注明出处:http://www.cnblogs.com/xsln/p/Introduction_TcMalloc.html 介绍: TcMalloc(Threa ...

    随机推荐

    1. linux编程进阶书推荐APUE,UNP

      编程进阶这里强烈推荐<unix环境高级编程>(简称APUE)和<unix网络编程>(简称UNP),这两本书可是经典中的经典啊,作 者是大名鼎鼎的 W.Richard Steve ...

    2. Cocos2d-x Tiled地图编辑器(一)基本使用

      Tiled地图编辑器支持普通视角地图和45度角地图, 它生成的地图数据文件cocos2d-x完美的支持,Tiled地图编辑器是一个以普通使用为目标地图编辑器,它使用简单而且能够轻松地在不同的游戏引擎中 ...

    3. How to debug with IntelliJ IDEA + Grails 2.3.x (转)

      问题: 最近访问grails.org,看到grails framework已经发展到2.3.x了,不免想尝尝鲜.下载了最新的grails-2.3.x之后,创建了一个新的grails app. 添加Bo ...

    4. java导入excel

      package com.duosen.gate.test; import java.io.File; import java.io.FileInputStream; import java.io.Fi ...

    5. Eureka 的 Application Client client的执行演示样例

              上篇以一个 demo 演示样例介绍了 Eureka 的 Application Service 客户端角色.今天我们继续了解 Eureka 的 Application Client 客 ...

    6. Codeforces Jzzhu and Sequences(圆形截面)

      # include <stdio.h> int f[10]; int main() { int x,y,n,j; while(~scanf("%d%d%d",& ...

    7. Windows Phone开发(1):概论

      原文:Windows Phone开发(1):概论 Windows Phone是微软公司开发的手机操作系统,这里就不多介绍,和Android,iPhone一样运行在智能手机上,相信大家都很熟悉. 目前来 ...

    8. Windows Phone开发(21):做一个简单的绘图板

      原文:Windows Phone开发(21):做一个简单的绘图板 其实我们今天要说的就是一个控件--InkPresenter,这个控件并不是十分强大,没办法和WPF中的InkCanvas相比,估计在实 ...

    9. unity3d插入Daikon Forge GUI 中国课程-7-高级控制slider采用

      (游戏开始的牛市)大家好我是孙广东.官方网站提供的是专业的视频教程http://www.daikonforge.com/dfgui/tutorials/,只是是在youtube上,要观看是须要FQ的. ...

    10. Nginx + Lua + redis (一)(转)

      使用 Lua 脚本语言操作 Redis. 由于大量的 Lua 代码写在 Nginx 中,会使配置文件显得很繁琐,所以这里使用 content_by_lua_file 来引入 Lua 脚本文件. 要使用 ...