一个简单的内存泄漏检测C工具
这个内存泄漏检测工具很简单,只能检测同一个模块,同一个线程中发送的内存泄漏,对于在编写代码过程中的代码调试有一定的帮助。如果要在集成测试或功能测试中检测内存泄漏,还需借助专门的工具。
1. 先取向malloc,free和calloc这几个标识符的定义:注意这一步非常重要,否则后面的malloc、free和calloc函数会和我们稍后在头文件中定义的宏冲突
// 取消malloc, calloc, free的宏定义 #undef malloc #undef calloc #undef free
2. 定义保存内存信息的单向链表
/**
* 定义链表节点,表示一个内存泄漏信息
*/
typedef struct _mem_node
{
void *ptr; // 泄漏内存地址
size_t block; // 泄漏内存大小
size_t line; // 泄露发生的代码行
char *filename; // 泄漏发生的文件名
struct _mem_node *next; // 下一个节点指针
} mem_node;
// 定义指向头节点的指针
mem_node *head = NULL;
3. 用于将节点加入单项链表的函数
/**
* 产生一个节点并加入链表
* @param ptr 分配的内存地址
* @param block 分配的内存单元大小
* @param line 代码行号
* @param filename 文件名称
*/
static void mem_node_add(void *ptr, size_t block, size_t line, char *filename)
{
// 产生节点
mem_node *node = malloc(sizeof(mem_node));
node->ptr = ptr;
node->block = block;
node->line = line;
node->filename = filename;
node->next = NULL;
// 加入链表头节点
if (head)
{
node->next = head;
head = node;
}
else
head = node;
}
4. 从单项链表中删除节点的函数
/**
* 从链表中删除一个节点
* @param ptr 分配的内存地址
*/
static void mem_node_remove(void *ptr)
{
// 判断头节点是否存在
if (head)
{
// 处理头节点
if (head->ptr == ptr)
{
// 获取头节点的下一个节点
mem_node *pn = head->next;
// 删除头节点
free(head);
// 令头节点指针指向下一个节点
head = pn;
}
else // 判断链表是否为空
{
// 指向节点的指针
mem_node *pn = head->next;
// 指向前一个节点的指针
mem_node *pc = head;
// 遍历所有节点
while (pn)
{
// 获取指向下一个节点的指针
mem_node *pnext = pn->next;
if (pn->ptr == ptr)
{
pc->next = pnext; // 删除当前节点
free(pn);
}
else
pc = pc->next;
pn = pnext;
}
}
}
}5. 显示内存泄露信息报告
/**
* 显示内存泄漏信息
*/
void show_block()
{
if (head)
{
// 保存总内存泄漏数量
size_t total = 0;
// 指向头节点的指针
mem_node *pn = head;
// 输出标题
puts("
-------------------------------内存泄漏报告------------------------------------
");
// 遍历链表
while (pn)
{
mem_node *pnext = pn->next;
// 处理文件名
char *pfile = pn->filename, *plast = pn->filename;
while (*pfile)
{
// 找到字符
if (*pfile == "\")
plast = pfile + 1; // 获取字符的位置
pfile++;
}
// 输出内存泄漏信息
printf("位置:%s(%d), 地址:%p(%dbyte)
", plast, pn->line, pn->ptr, pn->block);
// 累加内存泄漏总量
total += pn->block;
// 删除链表节点
free(pn);
// 指向下一个节点
pn = pnext;
}
printf("总计内存泄漏:%dbyte
", total);
}
}6. 定义调试用malloc函数
/**
* 用于调试的malloc函数
* @param elem_size 分配内存大小
* @param filename 文件名称
* @param line 代码行号
*/
void *dbg_malloc(size_t elem_size, char *filename, size_t line)
{
void *ptr = malloc(elem_size);
// 将分配内存的地址加入链表
mem_node_add(ptr, elem_size, line, filename);
return ptr;
}7. 定义调试用的calloc函数
/**
* 用于调试的calloc函数
* @param count 分配内存单元数量
* @param elem_size 每单元内存大小
* @param filename 文件名称
* @param line 代码行号
*/
void *dbg_calloc(size_t count, size_t elem_size, char *filename, size_t line)
{
void *ptr = calloc(count, elem_size);
// 将分配内存的地址加入链表
mem_node_add(ptr, elem_size * count, line, filename);
return ptr;
}8. 定义调试用的free函数
/**
* 用于调试的free函数
* @param ptr 要释放的内存地址
*/
void dbg_free(void *ptr)
{
free(ptr);
// 从链表中删除节点
mem_node_remove(ptr);
}上述代码应包含在一个C文件中(例如memcheck.c),完成上述步骤,就可以利用这一组函数来检测内存泄露了,需要定义如下头文件,该头文件应该被书写上述函数的C文件include:
#ifndef _MEM_CHECK_H #define _MEM_CHECK_H #include <stdlib.h> // instead of malloc #define malloc(s) dbg_malloc(s, __FILE__, __LINE__) // instead of calloc #define calloc(c, s) dbg_calloc(c, s, __FILE__, __LINE__) // instead of free #define free(p) dbg_free(p) /** * allocation memory */ void *dbg_malloc(size_t elem_size, char *filename, size_t line); /** * allocation and zero memory */ void *dbg_calloc(size_t count, size_t elem_size, char *filename, size_t line); /** * deallocate memory */ void dbg_free(void *ptr); /** * show memory leake report */ void show_block(); #endif // _MEM_CHECK_H
使用的时候只需要包含上述头文件(例如命名为memcheck.h),并将上述C文件引入到项目中即可。测试代码如下:
#ifdef DEBUG
#include "memcheck.h"
#endif
int main()
{
int* p;
#ifdef DEBUG
atexit(show_block); // 在程序结束后显示内存泄漏报告
#endif // DEBUG
// 分配内存并不回收,显示内存泄漏报告
p = (int*)malloc(1000);
return 0;
}声明:该文观点仅代表作者本人,入门客AI创业平台信息发布平台仅提供信息存储空间服务,如有疑问请联系rumenke@qq.com。
- 上一篇:没有了
- 下一篇:没有了
