valgrind 进阶-2


支持原创,转载请附上原文链接


0 引言

前面介绍了valgrind的基础,参见 Valgrind 基础,今天这结合Valgrind,来讲讲内存泄漏,以及基于Valgrind定位、分析内存泄漏的罪魁祸首

索引:
1、valgrind 实战
2、”still reachable“ 是否关注
3、内存泄漏的思考

1 内存泄漏与 valgrind

1.1 什么是内存泄漏?

内存泄漏(Memory Leak)是指程序中已动态分配的堆内存由于某种原因程序未释放或无法释放,造成系统内存的浪费,导致程序运行速度减慢甚至系统崩溃等严重后果

内存泄漏-百科
用程序员的话讲:就是malloc 或者 new 了分配了内存空间,由于某种原因没有执行free 或者delete释放内存空间,导致系统内存逐渐变少以至于无内存可用

1.2 理论 && 实际

关于一些比较明确的内存泄漏案例,我这里就简单带过,比如下面代码

#include <iostream>
void leak(){
  char* p = new char[1024];
}
int main() {
  leak();
  return 0;
}

通过valgrind工具运行检查,结果如下

==11562== HEAP SUMMARY:
==11562==     in use at exit: 73,728 bytes in 2 blocks
==11562==   total heap usage: 2 allocs, 0 frees, 73,728 bytes allocated
==11562==
==11562== 1,024 bytes in 1 blocks are definitely lost in loss record 1 of 2
==11562==    at 0x4C2E80F: operator new[](unsigned long) (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==11562==    by 0x400717: leak() (in /home/a.out)
==11562==    by 0x400727: main (in /home/a.out)
==11562==
==11562== 72,704 bytes in 1 blocks are still reachable in loss record 2 of 2
==11562==    at 0x4C2DB8F: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==11562==    by 0x4EC3EFF: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==11562==    by 0x40106F9: call_init.part.0 (dl-init.c:72)
==11562==    by 0x401080A: call_init (dl-init.c:30)
==11562==    by 0x401080A: _dl_init (dl-init.c:120)
==11562==    by 0x4000C69: ??? (in /lib/x86_64-linux-gnu/ld-2.23.so)
==11562==
==11562== LEAK SUMMARY:
==11562==    definitely lost: 1,024 bytes in 1 blocks
==11562==    indirectly lost: 0 bytes in 0 blocks
==11562==      possibly lost: 0 bytes in 0 blocks
==11562==    still reachable: 72,704 bytes in 1 blocks
==11562==         suppressed: 0 bytes in 0 blocks

很明显,leak()函数内存泄露了,valgrind的检测结果也明确指出:definitely lost: 1,024 bytes in 1 blocks

1.2.1 似而不是的内存泄漏

? 那么我首先提一个问题:是否所有的malloc/new没有对应的free/delete,都是内存泄漏,需要我们修复呢?

其实,这个问题并不难,仔细看内存泄漏的定义也可以得出答案,但是这里,我通过一段代码,来给大家看看,加深理解

#include <iostream>
struct GlobalConf{
  int conf1;
  int conf2;
  int conf3;
};
GlobalConf* conf2;
int main() {
  conf2 = new GlobalConf();
  conf2->conf1 = 1;
  conf2->conf2 = 2;
  conf2->conf3 = 3;

  /*
   * 整个生命周期conf2都需要使用
   */
  while(1);
  return 0;
}

上面我们有一个全局配置,conf2指针,程序初始化的时候,通过分配空间的方式分配内存,该程序一直运行不会结束,conf2虽然分配了内存,但是其实只是在初始化的时候分配一次,且整个程序声明周期一直在使用,所以虽然看起来分配了内存未释放,但是其实并不属于内存泄漏

1.2.2 引发血案的内存泄漏

内存泄漏之所以比较难查,一个原因就是不同于数组越界等其他的内存非法访问,存在很大的隐蔽性,而且有时候,valgrind提供的信息,还需要结合我们的代码,才能真正定位问题,下面,我分析一个案例,讲述still reachable在我们分析问题时候的重要性,不可忽视,特别是blocks比较多,一定需要关注

#include <unistd.h>
#include <iostream>
#include <thread>
#include <vector>
class Buffer {
 public:
  Buffer(int size) : size_(size) { buf_ = new char[size]; }
  ~Buffer() {
    size_ = 0;
    delete[] buf_;
  }
 private:
  char* buf_;
  int size_;
};
std::vector<std::shared_ptr<Buffer>> vectors_;
std::vector<std::shared_ptr<Buffer>> vectors2_;
void t1_handler() {
  while (1) {
    auto item = std::make_shared<Buffer>(1024);
    vectors_.emplace_back(item);
    auto item2 = std::make_shared<Buffer>(1024);
    vectors2_.emplace_back(item2);
    std::cout << "insert" << std::endl;
    sleep(2);
  }
}
void t2_handler() {
  while (1) {
    if (!vectors_.empty()) {
      std::cout << "before --> size:" << vectors_.size() << std::endl;
      vectors_.clear();
      std::cout << "after --> size:" << vectors_.size() << std::endl;
      std::cout << "vector2 --> size:" << vectors2_.size() << std::endl;
      sleep(1);
    }
  }
}
int main() {
  std::thread t1(t1_handler);
  std::thread t2(t2_handler);
  t1.detach();
  t2.detach();

  while (1)
    ;
  return 0;
}

通过valgrind执行程序的日志如下:

==18654== HEAP SUMMARY:
==18654==     in use at exit: 87,376 bytes in 33 blocks
==18654==   total heap usage: 60 allocs, 27 frees, 100,256 bytes allocated
==18654==

省略...

==18654== 12,288 bytes in 12 blocks are still reachable in loss record 10 of 11
==18654==    at 0x4C2E80F: operator new[](unsigned long) (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==18654==    by 0x40170F: Buffer::Buffer(int) (in /home/a.out)
==18654==    by 0x4037F7: _ZN9__gnu_cxx13new_allocatorI6BufferE9constructIS1_IiEEEvPT_DpOT0_ (in /home/a.out)
==18654==    by 0x4034CD: _ZNSt16allocator_traitsISaI6BufferEE9constructIS0_IiEEEvRS1_PT_DpOT0_ (in /home/a.out)
==18654==    by 0x403123: std::_Sp_counted_ptr_inplace<Buffer, std::allocator<Buffer>, (__gnu_cxx::_Lock_policy)2>::_Sp_counted_ptr_inplace<int>(std::allocator<Buffer>, int&&) (in /home/a.out)
==18654==    by 0x402C86: std::__shared_count<(__gnu_cxx::_Lock_policy)2>::__shared_count<Buffer, std::allocator<Buffer>, int>(std::_Sp_make_shared_tag, Buffer*, std::allocator<Buffer> const&, int&&) (in /home/a.out)
==18654==    by 0x402915: std::__shared_ptr<Buffer, (__gnu_cxx::_Lock_policy)2>::__shared_ptr<std::allocator<Buffer>, int>(std::_Sp_make_shared_tag, std::allocator<Buffer> const&, int&&) (in /home/a.out)
==18654==    by 0x402397: std::shared_ptr<Buffer>::shared_ptr<std::allocator<Buffer>, int>(std::_Sp_make_shared_tag, std::allocator<Buffer> const&, int&&) (in /home/a.out)
==18654==    by 0x401CB3: std::shared_ptr<Buffer> std::allocate_shared<Buffer, std::allocator<Buffer>, int>(std::allocator<Buffer> const&, int&&) (in /home/a.out)
==18654==    by 0x401833: _ZSt11make_sharedI6BufferIiEESt10shared_ptrIT_EDpOT0_ (in /home/a.out)
==18654==    by 0x4012AF: t1_handler() (in /home/a.out)
==18654==    by 0x404358: void std::_Bind_simple<void (*())()>::_M_invoke<>(std::_Index_tuple<>) (in /home/a.out)
==18654==
==18654== 72,704 bytes in 1 blocks are still reachable in loss record 11 of 11
==18654==    at 0x4C2DB8F: malloc (in /usr/lib/valgrind/vgpreload_memcheck-amd64-linux.so)
==18654==    by 0x50E0EFF: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==18654==    by 0x40106F9: call_init.part.0 (dl-init.c:72)
==18654==    by 0x401080A: call_init (dl-init.c:30)
==18654==    by 0x401080A: _dl_init (dl-init.c:120)
==18654==    by 0x4000C69: ??? (in /lib/x86_64-linux-gnu/ld-2.23.so)
==18654==
==18654== LEAK SUMMARY:
==18654==    definitely lost: 0 bytes in 0 blocks
==18654==    indirectly lost: 0 bytes in 0 blocks
==18654==      possibly lost: 576 bytes in 2 blocks
==18654==    still reachable: 86,800 bytes in 31 blocks
==18654==         suppressed: 0 bytes in 0 blocks
==18654==

valgrind 是在程序结束的时候检测,上面程序while运行不能结束,因此需要ctrl+c结束程序

上面still reachable显示有12blocks泄漏了12M的数据,位置是在t1_handler中分配的。
上面看起来可能一眼能看出来,vectors2_没被清理,那是因为这是一个简单的测试demo,显示中这种隐蔽的case可能就比较难查了


关于valgrind的经验总结:

  1. definitely lost 需要全部清理掉
  2. still reachable 需要关注,但是需要case by case的分析,特别是有多个blocks泄漏时,表明内存泄漏位置被重复调用了,这种一般会在长时间运行后由于无内存可用导致系统崩溃
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容