C 内存管理模块的编写

C语言手动管理内存很困难,有时候很难发现内存泄漏,这两天看了一下<自制编程语言>里面有写了一个简单的内存管理模块,发现挺精巧,可以有效检测内存泄漏

原理很简单,就是把C的malloc函数分配的内存用一个链表记录起来,最后查看这个链表是否还有节点就可以判断是否有内存泄漏了

首先我们先看一下这个链表的数据结构

//标记数组的大小
#define MARK_SIZE 4
typedef struct Mem_Header Mem_Header;
struct Mem_Header
{
int size;//后面内存分配的大小
Mem_Header *next;//下一个节点的指针
Mem_Header *prev;//上一个节点的指针
char *filename;//文件名指针
int line;//在第几行
unsigned char mark[MARK_SIZE];//标记数组,这个数据用来表示这块内存是否被破坏了
}

这个mark数组我们将用 0xcd填充,如果这个数组被动过了就说明这块内存被破坏了

我们每次malloc一块内存的时候,这个结构体就会被添加到这块内存的头部

就像这样:

void *p = malloc(size);
Mem_Header *header = (Mem_Header *)p;
//然后设置header信息
/*省略一些赋值过程*/
return (char*)header+sizeof(Mem_Header);

这样外部拿到的指针就可以随便用了,回收的时候还可以重新推出头部指针的信息,然后从链表中删除这个节点

为了防止后面溢出,可以在后面内存也加上 mark数组 这样我们每次分配的内存大小是 sizeof(Mem_Header) + size + MARK_SIZE其中size是需要分配的大小,MARK_SIZE是在后面加的标记数组

我们用malloc初始化的时候经常碰到0,我们把这块内存填充0xcc(没意义的值别的其实也行),更容易发现错误

memset(p, 0xcc, willAlloc);

我们先来看第一个函数:

//size 需要分配的大小
//filename 文件名
//line 行好
void *Mem_alloc(Mem_Controller *controller, size_t size, char *filename, int line)
{
int willAlloc = size + MEM_HEADER_SIZE + MARK_SIZE;//实际分配大小
void *p = malloc(willAlloc);
if (!p)
controller->handle(controller, filename, line, "malloc null");
//填充数组
memset(p, 0xcc, willAlloc);
//设置头部一些信息
Mem_setHeader(p, size, filename, line);
//设置tail的信息
Mem_setTail(p, size);
//把这个节点添加到链表中
Mem_addChain(controller, (Mem_Header *)p);
return (char *)p + MEM_HEADER_SIZE;
}

这个函数就是分配一个内存,controller是控制器,实际上就是负责打印和保存头指针的

filename和line就是分配空间时候的文件和行号,可以用宏定义解决可以百度一下这两个宏__LINE__,__FILE__

内存释放:从链表中删除这个节点

void Mem_free(Mem_Controller *controller, void *p, char *filename, int line)
{
if (!p)
return;
//头信息
Mem_Header *header = (char *)p - MEM_HEADER_SIZE;
//check header and tail
//检查是不是mark数组损坏了
Mem_check(controller, header, filename, line);
//移除节点
Mem_rmChain(controller, header);
free(header);
}

打印所有链表中的节点

void Mem_dumps(Mem_Controller *controller)
{
Mem_Header *pos = controller->memHeader;
FILE *fp = stderr;
int counter = 0;
for (pos = controller->memHeader; pos; pos = pos->next)
{
Mem_check(controller, pos, pos->filename, pos->line);
fprintf(fp, "[%04d]%p********************\n", counter,
(char *)pos + MEM_HEADER_SIZE);
fprintf(fp, "%s line %d size..%d\n",
pos->filename, pos->line, pos->size);
counter++;
}
}

为了方便我们使用,我们可以设置几个宏定义

typedef struct Mem_Controller Mem_Controller;
extern Mem_Controller *pcontrol; void *Mem_alloc(Mem_Controller *controller, size_t size, char *filename, int line);
void Mem_free(Mem_Controller *controller, void *p, char *filename, int line);
void Mem_dumps(Mem_Controller *controller); #define current_mem_controller pcontrol
#define Mem_alloc_func(size) Mem_alloc(current_mem_controller, size, __FILE__, __LINE__)
#define Mem_free_func(p) Mem_free(current_mem_controller, p, __FILE__, __LINE__)
#define Mem_dump_func() Mem_dumps(current_mem_controller)

那个extern全局变量是别的文件定义的指针,__FILE__会自动替换文件名,这样我们就可以看到是哪里的内存没有被释放

使用的时候也简单了很多直接用宏定义就可以了

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include "MEM.h" void fill_buffer(unsigned char *buf, int size)
{
int i; for (i = 0; i < size; i++)
{
buf[i] = i;
}
}
void mem_print(unsigned char *p, int size)
{
for (int i = 0; i < size; i++)
{
fprintf(stderr, "%02x", p[i]);
}
fprintf(stderr, "\n");
} int main()
{
void *p = Mem_alloc_func(10);
mem_print(p, 10);
void *p2 = Mem_alloc_func(10);
mem_print(p2, 10);
void *p3 = Mem_alloc_func(10);
mem_print(p3, 10);
Mem_dump_func();
fprintf(stderr, "try to clear\n");
Mem_free_func(p2);
Mem_free_func(p);
Mem_free_func(p3);
Mem_dump_func();
fprintf(stderr, "then to malloc\n");
p = Mem_alloc_func(10);
fill_buffer(p, 10);
p = Mem_realloc_func(p, 20);
mem_print(p, 24);
Mem_dump_func();
return 0;
}

我们可以看到泄漏的大小和行号

原版的代码的联合体是用来内存对齐的加快访问速度

一个C语言内存管理模块的实现的更多相关文章

  1. Go语言内存管理(一)内存分配

    Go语言内存管理(一)内存分配 golang作为一种"高级语言",也提供了自己的内存管理机制.这样一方面可以简化编码的流程,降低因内存使用导致出现问题的频率(C语言使用者尤其是初学 ...

  2. C语言 内存管理(转)

     转自 https://blog.csdn.net/u011616739/article/details/61621815 C语言 内存管理 1.内存分区 C源代码进过预处理.编译.汇编和链接4步生成 ...

  3. 深入理解Linux C语言内存管理

    问题不能拖,我这就来学习一下吧,争取一次搞定. 在任何程序设计环境及语言中,内存管理都十分重要. 内存管理的基本概念 分析C语言内存的分布先从Linux下可执行的C程序入手.现在有一个简单的C源程序h ...

  4. rt-thread中动态内存分配之小内存管理模块方法的一点理解

    @2019-01-18 [小记] rt-thread中动态内存分配之小内存管理模块方法的一点理解 > 内存初始化后的布局示意 lfree指向内存空闲区首地址 /** * @ingroup Sys ...

  5. C语言内存管理(转)

    伟大的Bill Gates 曾经失言: 640K ought to be enough for everybody — Bill Gates 1981 程序员们经常编写内存管理程序,往往提心吊胆.如果 ...

  6. iOS学习08之C语言内存管理

    本次主要学习和理解C语言中的内存管理 1.存储区划分 按照地址从高到低的顺序:栈区,堆区,静态区,常量区,代码区 1> 栈区:局部变量的存储区域 局部变量基本都在函数.循环.分支中定义 栈区的内 ...

  7. 深入理解C语言内存管理

    之前在学Java的时候对于Java虚拟机中的内存分布有一定的了解,但是最近在看一些C,发现居然自己对于C语言的内存分配了解的太少. 问题不能拖,我这就来学习一下吧,争取一次搞定. 在任何程序设计环境及 ...

  8. Objective-C语言内存管理

    • Objective-C为每个对象提供一个内部计数器,这个计数器跟踪对象的引用次数.所有类都继承自 NSObject 的对象retain和release方法. 如果使用了new.alloc或copy ...

  9. 黑马程序员——OC语言 内存管理

    Java培训.Android培训.iOS培训..Net培训.期待与您交流! (以下内容是对黑马苹果入学视频的个人知识点总结) (一)计数器 每个对象内部都保存了一个与之相关联的整数,称为引用计数器,当 ...

随机推荐

  1. redis 事务、Jedis事务处理流程

    127.0.0.1:6379> multiOK127.0.0.1:6379> sadd myset a b cQUEUED  ——>并没有执行,排队等待127.0.0.1:6379& ...

  2. 分析VoltDB内存数据库

    转自https://blog.csdn.net/olidrop/article/details/7065384 https://blog.csdn.net/ransom0512/article/det ...

  3. linux中ftp提示--553 Could not create file

    今天在阿里云的linux上搭建ftp服务的时候,搭建成功之后,上传文件时总提示553 Could not create file,找了半天原因,终于解决了 ftp主目录为/home/myftp /ho ...

  4. 线段树&&线段树的创建线段树的查询&&单节点更新&&区间更新

    目录 线段树 什么是线段树? 线段树的创建 线段树的查询 单节点更新 区间更新 未完待续 线段树 实现问题:常用于求数组区间最小值 时间复杂度:(1).建树复杂度:nlogn.(2).线段树算法复杂度 ...

  5. 【接口】常见接口集合(返回JSON)

    转<JSON校验网站…>http://www.bejson.com/go.html?u=http://www.bejson.com/webInterface.html 这里为大家搜集了一些 ...

  6. HTML、jsp页面中radio,checkbox,select数据回显功能,默认被选中问题

    最近常常遇到各种复选框.单选框.下拉框的默认被选中的问题,开始也是绞尽脑汁的想办法,今天写一篇学习总结的博文来写一下学习总结. 单选框(radio)默认被选中: 一.jstl技术进行回显 <in ...

  7. Jmeter入门3 http请求—content-type与参数

    本文讲三种content-type以及在Jmeter中对应的参数输入方式 第一部分:目前工作中涉及到的content-type 有三种: content-type:在Request Headers里, ...

  8. ajax跨域请求在IE8中存在的问题

    从没打算怎么去兼容老版本IE,毕竟微软自己都放弃了,可是最近做好的东西在所有的IE下都会出问题:GetJson不执行 本来觉得挺简单的,度娘也给出了一大堆的解决方案,可惜,基本上都是在说缓存,实际上并 ...

  9. Python中网络编程对 listen 函数的理解

    listen函数的第一个参数时SOCKET类型的,该函数的作用是在这个SOCKET句柄上建立监听,至于有没有客户端连接进来,就需要accept函数去进行检查了,accept函数的第一个参数也是SOCK ...

  10. AOP-方法拦截器-笔记

    方法拦截器的继承层次图: 这些拦截器具体长什么样?? 一.MethodBeforeAdviceInterceptor 这个拦截器只有一个属性就是前置通知.需要注意的是前置通知和返回通知的拦截器才会持有 ...