malloc,free简单的实现
malloc函数分配内存时会附加若干(一般是12个)字节,存放控制信息。
该信息一旦被意外损坏,可能在兴许操作中引发异常。
brk/sbrk 底层维护一个指针,记录所分配的内存结尾,所分配内存位于堆中,底层调用mmap/munmap。
malloc 底层维护一个双向链表和必要的控制信息,不可越界訪问。所分配内存位于堆中,底层调用brk/sbrk。
每一个进程都有4G的虚拟内存空间。虚拟内存地址仅仅是一个数字,并没有和实际的物理内存相关联。
所谓内存分配与释放,其本质就是建立或取消虚拟内存和物理内存间的映射关系。
sbrk (
intptr_t increment // 内存增量(以字节为单位)
);
0 - 获取末尾地址。
>0 - 添加内存空间。
<0 - 释放内存空间。
内部维护一个指针。指向当前堆内存最后一个字节的下一个位置。
sbrk函数依据增量參数调整该指针的位置,同一时候返回该指针原来的位置。
若发现页耗尽或空暇。则自己主动追加或取消页映射。
brk (
);
成功返回0。失败返回-1。
内部维护一个指针,指向当前堆内存最后一个字节的下一个位置。
brk函数依据指针參数设置该指针的位置。
若发现页耗尽或空暇。则自己主动追加或取消页映射。
简便起见,可用sbrk分配内存。用brk释放内存。
#include <stdio.h>
#include <unistd.h>
#include <stdbool.h>
typedef struct mem_control_block //内存控制块
{
bool free; //自由标志
struct mem_control_block* prev; //前向指针
size_t size; //块大小
}MCB;
MCB* g_top; //内存栈顶指针 // +----------------------+ g_top
// v | |
// +------+------------+--|---+------------+------+------------+
// | prev | | prev | | prev | |
// | free | | free | | free | |
// | size | | size | | size | |
// +------+------------+------+------------+------+------------+
// MCB |<-- size -->| void* my_malloc(size_t size)
{
MCB* mcb;
for(mcb = g_top; mcb; mcb = mcb->prev)
if(mcb->free && mcb->size >= size)//寻找可用空块
break;
if(!mcb) //假设没有可用空块
{
mcb = sbrk(sizeof(MCB) + size); //增量分配size大小内存
if(mcb == (void*)-1) //假设分配失败,打印错误信息
{
perror("sbrk");
return NULL;
}
mcb->prev = g_top; //调整前向指针
mcb->size = size; //大小
g_top = mcb; //调整栈顶指针
}
mcb->free = false; //不管用原来空块还是新分配的块,都标记为不可用
return mcb + 1; //返回实际分配的内存起始地址
}
void my_free(void* ptr)
{
if(!ptr)
return; MCB* mcb = (MCB*)ptr - 1; //取控制块起始地址,
//注意:ptr为实际可用内存起始地址
mcb->free = true; //块标记为可用 //在栈中查找连续内存块
for(mcb = g_top; mcb->prev; mcb = mcb->prev)
if(!mcb->free)
break;
//释放整个栈全部内存块
if(mcb->free)
{
g_top = mcb->prev;
brk(mcb); //改动内存块末尾地址
}
else //释放连续的标记为true的内存块
{
g_top = mcb;
brk((void*)mcb + sizeof(MCB) + mcb->size);
}
}
//測试
int main(void)
{
int* pa[10];
size_t size = sizeof(pa) / sizeof(pa[0]), i, j;
for(i = 0; i < size; ++i)
{
if(!(pa[i] = (int*)my_malloc((i+1) * sizeof(int))))
{
perror("my_malloc");
return -1;
}
for(j = 0; j <= i; ++j)
pa[i][j] = j;
}
for(i = 0; i < size; ++i)
{
for(j = 0; j <= i; ++j)
printf("%d ", pa[i][j]);
printf("\n");
}
for(;;)
{
my_free(pa[--i]);
if(! i)
break;
}
return 0;
}
版权声明:本文博主原创文章,博客,未经同意不得转载。
malloc,free简单的实现的更多相关文章
- new与malloc
首先将new与malloc的区别总结如下: 1 new可以自动计算需要分配多大的内存,而malloc必须指明. 2 new返回的指针是有类型的,malloc返回void*类型的指针. 3 new在分配 ...
- 简单来谈谈alloc分配器
之前说道alloc是原G2.9版本的默认的分配器,这篇就把alloc的原理梳理梳理,顺便简单介绍下有关的内存管理. 一般而言,我们通常习惯的内存分配操作和释放操作是这样的: class Foo {…} ...
- 从内部入手,浅谈malloc和new的区别
想要理解一样事物,就要先用自己的语言去描述一件事物.在我查阅资料后,发现malloc函数简单说来就是空闲内存空间收集器,并把空闲空间关联起来,用术语来说就是:将空闲内存块合并起来并称为"闲置 ...
- C语言内存分配函数malloc——————【Badboy】
C语言中经常使用的内存分配函数有malloc.calloc和realloc等三个,当中.最经常使用的肯定是malloc,这里简单说一下这三者的差别和联系. 1.声明 这三个函数都在stdlib.h库文 ...
- new/delete与malloc/free的区别
new/delete与malloc/free的区别 参考: https://blog.csdn.net/u013485792/article/details/51329541 https://www. ...
- malloc实现机制
使用过c语言的都知道malloc是一个动态分配内存的函数,还可以通过free释放内存空间. 如果我们想分析一下malloc的源码,这其实不是一会就能看懂的,但是我们可以讨论一下malloc的简单实现. ...
- STL—内存的配置与释放
上一篇我们介绍了STL对象的构造与析构,这篇介绍STL内存的配置与释放. STL有两级空间配置器,默认是使用第二级.第二级空间配置器会在某些情况下去调用第一级空间配置器.空间配置器都是在allocat ...
- [自制操作系统] 连续页分配释放&kmalloc/kfree
本文将在JOS上实现连续内存.释放,提供内核的kmalloc与kfree,并在分配frambuffer的时候进行测试. Github : https://github.com/He11oLiu/JOS ...
- STL源码剖析(空间配置器)
前言 在STL中,容器的定义中都带一个模板参数,如vector template <class T, class Alloc = alloc> class vector {...} 其中第 ...
随机推荐
- HashMap的遍历和排序
1.HashMap的遍历 package com.sheepmu; import java.util.HashMap; import java.util.Iterator; import java.u ...
- 自己实现的Boost库中的lexical_cast随意类型转换
知道了C++的I/O设施之后.这些就变的非常easy了. 假设你常常使用,时间长了就会有感觉.这个事情是多此一举吗?就当是练习吧,知道原理之后,你会认为用起来更舒畅,更喜欢C++了. #include ...
- 全文检索引擎Solr 指南
全文检索引擎Solr系列:第一篇:http://t.cn/RP004gl.第二篇:http://t.cn/RPHDjk7 .第三篇:http://t.cn/RPuJt3T
- 使用 WPF 创建单实例应用程序
一个简单的例子就是大家在使用很多应用程序,例如在使用Microsoft Word 时会遇到一种情况,不管你打开多少个文档,系统一次只能加载一个winword.exe 实例.当打开新文档时,文档在新窗口 ...
- android数据访问模式:档、SharedPreferences
android数据访问模式:档.SharedPreferences.SQLite 数据库.Content provider 文件流: 使用java IO流对文件进行读写操作,文件权限默认. 指定文件权 ...
- linux如何执行后台进程
linux直接执行一个过程.电流指令结束后.或者关闭掉shell形成过程将结束. 如何在后台执行的处理 办法1 采用nohup命令,nohup命令本身的意思no hung up他说,他们将不会收到sh ...
- ZOJ3827 ACM-ICPC 2014 亚洲区域赛的比赛现场牡丹江I称号 Information Entropy 水的问题
Information Entropy Time Limit: 2 Seconds Memory Limit: 131072 KB Special Judge Informatio ...
- 流动python - 什么是魔术方法(magic method)
我们经常看到各种各样的方法已经被包围了由双下划线,例如__init__,他们是魔术方法. 魔术方法python语言预订好"协议",在不同情况下不同的魔术方法,是隐式调用.我们重写这 ...
- unity3d中让物体显示和隐藏
unity3d中让物体显示和隐藏的方法 gameObject.renderer.enabled //是控制一个物体是否在屏幕上渲染或显示 而物体实际还是存在的 仅仅是想当于隐身 而物体本身的碰撞体还 ...
- 对比Windows 8模拟器(Simulator)和Windows Phone仿真器(Emulator)
原文:对比Windows 8模拟器(Simulator)和Windows Phone仿真器(Emulator) 从事移动应用开发,经常会用到模拟器(Simulator)和仿真器(Emulator),本 ...