> 文档中心 > 一个C++工程内存泄漏问题的排查及重现工程

一个C++工程内存泄漏问题的排查及重现工程

最近遇到一个C++工程内存泄漏的问题,经过排查,发现原来是 map 的使用有问题,本文记录了排查的过程,并给出一个类似的工程代码。

起因

某日,运维反馈生产环境某台设备出现问题,经组长排查,有两个工程服务占用内存较多,出现 OOM 被 Linux 系统干掉了。其中一个是我接手的工程,竟达到了 6GB,随即安排我排查。

排查

首先在本地虚拟机用 cppcheck、valgrind 测试,但没有发现容易看得懂的问题点,像 cppcheck 提示了很多不怎么要紧的问题——其实有大半问题已经在前两个月修正了。而 valgrind提示多的都是第三方库,比如 curl、xml、ssl 等。

因为没有头绪,也不敢随便动生产环境,所以写了个简单的 shell 脚本,用于监控程序的内存使用情况,并放在生产环境上,观察半天,发现隔1分钟就有少量内存泄漏,大概几十 KB 左右。因此得到存在内存泄漏的结论,但这只是验证猜测而已,因为在问题发现之初就已经把问题引致这方面了。

由于代码年代久远,错综复杂,几天过去也没头绪,还好发现概率比较小,还有时间排查。

后经同事指点,将监控程序频率提高,输出内存的同时打印日期时间,将其与工程日志的日期对比,缩小可疑范围,最后定位到传输模块的一个函数

该函数使用 malloc 根据某个数据表名称为一个结构体变量指针申请内存,再放到 map 全局变量中,由于外部函数使用到,故不能释放,跟踪发现在类的析构函数中会释放内存,但在程序运行过程并没有进行析构,所以一直没有释放内存。存放到 map 的目的是防止多次申请内存,因为数据表的数量有限——不到十个,因此使用 map,在申请之前会查找 map,如不存在再申请,并存起来。

业务逻辑上并无问题,后在某个不起眼的地方看到了对该 map 变量的清除操作,即调用 clear 函数。怀疑此函数使用有误,于是写了一个简单的测试程序重现问题。最终得到结论:调用 map 的 clear 函数会清除 key,但如果 key 为指针,则不会释放其指向的内存。这正是问题根本原因所在。

重现问题

用于重现问题的测试程序如下:

#include #include #include #include #include typedef struct {    char Name[1024];    char Name1[1024];    char Name3[1024];} TTableStruct;class CMapLeak {public:    CMapLeak();    ~CMapLeak(); TTableStruct *GetTable(const char *TableName);    void TableTest();private:    std::map < std::string, TTableStruct * >m_mTable;};CMapLeak::CMapLeak(){    }CMapLeak::~CMapLeak(){    std::map < std::string, TTableStruct * >::iterator iter;    for (iter = m_mTable.begin(); iter != m_mTable.end(); iter++)    { TTableStruct *pStruct = iter->second; if (pStruct != NULL) {   delete pStruct; }    }}TTableStruct* CMapLeak::GetTable(const char *TableName){    TTableStruct *pStruct = NULL;    std::map < std::string, TTableStruct * >::iterator iter;    iter = m_mTable.find(TableName);    if (iter == m_mTable.end())    { pStruct = new TTableStruct[100]; m_mTable[TableName] = pStruct; printf("NEW!!! struct ptr: %p\n", pStruct);    }    else    { pStruct = iter->second; printf("struct ptr: %p\n", pStruct);    }    return pStruct;}void CMapLeak::TableTest(){    TTableStruct *pStruct = NULL;    int i = 0;    char tablename[32] = {0};    while (1)    { //m_mTable.clear(); // !!! 如执行此行,则会清空 map 的 key sprintf(tablename, "table_%d", (i++)&0x03); pStruct = GetTable(tablename); printf("%s: struct ptr: %p\n", tablename, pStruct); printf("----------------\n"); sleep(1);    }}int main(void){    CMapLeak* pLeak = new CMapLeak();    pLeak->TableTest();    return 0;}

代码逻辑比较简单,为模拟生产环境的运行,直接使用死循环执行。先查找 m_mTable,如果 key 不存在则申请内存,否则直接返回已申请的内存。为了方便观察内存使用情况,在结构体中多加了几个数组。

当对 map 进行 clear 操作时,出现内存泄漏,监控脚本输出如下:

有内存泄漏的:23:14:31dataserver ps mem: 13596 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413468 kB Cached: 637216 kB-------------23:14:36dataserver ps mem: 15116 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2412884 kB Cached: 637216 kB-------------23:14:41dataserver ps mem: 16636 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413492 kB Cached: 637216 kB-------------23:14:46dataserver ps mem: 18460 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413144 kB Cached: 637216 kB-------------23:14:52dataserver ps mem: 19980 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2412740 kB Cached: 637216 kB-------------23:14:57dataserver ps mem: 21500 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413104 kB Cached: 637216 kB-------------23:15:02dataserver ps mem: 23020 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413364 kB Cached: 637216 kB-------------

如果不调用 clear 函数,则内存占用较稳定:

23:10:12dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413616 kB Cached: 637212 kB-------------23:10:17dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2412888 kB Cached: 637212 kB-------------23:10:22dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413504 kB Cached: 637212 kB-------------23:10:27dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413092 kB Cached: 637212 kB-------------23:10:32dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413536 kB Cached: 637212 kB-------------23:10:37dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2412988 kB Cached: 637212 kB-------------23:10:43dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413348 kB Cached: 637212 kB-------------23:10:48dataserver ps mem: 13900 VmRSS: 1068 kBSystem memory info:  MemTotal: 3861496 kB MemFree: 2413796 kB Cached: 637212 kB-------------

小结

就目前排查结果看,只需要将原工程清除 map 的 clear 函数去掉即可。但排查过程,还是花了一定的时间。