这一篇主要介绍一下C中的malloc和free(当然在C++中它们也可以使用),下一篇将主要介绍一下C++中的new和delete
如有侵权,请联系删除,如有错误,欢迎大家指正,谢谢

0. malloc

基本用法

  • 首先malloc()是程序员在堆区申请空间的函数,原型为void *malloc(size_t size); size是要在堆区申请的字节数,它的返回值是申请空间的首地址,它申请的空间是连续的
  • size_t数据类型经常用到,在 32bit 编译器中是unsigned int;在64 bit系统中是unsigned __int64
  • malloc(大于0); 如可用内存不足,返回空指针NULL,如可用内存足够,返回申请空间的首地址;malloc(0)返回一个有效的空间长度为零的内存首地址,但是没法用(只进行申请和释放可以,如申请后执行了写操作,释放时会报错);
// MSDN 推荐使用方法
int* n = (int*)malloc(sizeof(int)); // C++ void*向其他类型的隐式转换需进行强制类型转换,C语言void*向其他类型的隐式转换,可不进行强制类型转换
if (n == NULL){// 进行内存申请失败的错误处理
} else {// 进行正常操作free(n);
}

动态申请数组

  • 申请一个有6个整形数组元素的一维数组,申请完不能初始化(代码如下),只能通过memset()或循环的方式赋值
int* p = (int*)malloc(sizeof(int) * 6);
  • 因该操作程序运行到这条语句时才在堆区申请的数组,所以被称为动态申请内存(数组),栈区的数组在编译时就已经申请好内存了,所以不是动态申请的

动态申请数组指针

int (*p)[3] = (int(*)[3])malloc(sizeof(int) * 3); // 一维数组指针
int (*q)[2][3] = (int(*)[2][3])malloc(sizeof(int) * 6); // 二维数组指针

初始化

void* memset(void* dest, int c, size_t count);
  • 此函数是按字节进行赋值的
  • dest指向目标地址的指针;c要赋值给内存空间的值;count是字节数;返回值是被赋值的目标地址的指针
void* memcpy(void* dest, void* src, size_t count);
  • 此函数也是按照字节进行拷贝的
  • dest指向目标地址的指针,也就是要被赋值的空间首地址;src指向源地址的指针,也就是要被复制的空间的首地址;count跟memset()一样表示被拷贝的字节数;返回值也是被赋值的目标地址的指针

1. 其他申请内存的方式

calloc

void* calloc(size_t num, size_t size);
  • 申请连续的 num 块内存,每块内存的字节数为size;并将这些字节置为初始化为0,返回值为所申请空间的首地址,申请数组时比较方便,但是效率可能比malloc()会慢一点,因为多了一步初始化操作

realloc

void* realloc(void* memblock, size_t size); // 为已分配的内存空间重新申请内存块
  • memblock指向之前已分配内存块的指针;size新内存块大小(字节数);返回值是重新分配内存块的首地址
  • 如果原来分配的内存块的地方无法再扩展到size要求的大小,那么会重新分配一块size大小的内存,原地址的内容会被拷贝过去,相应的返回值也会是新分配区域的首地址,如果可以扩展到指定大小,那返回值还会是重新分配前的返回值

_msize

size_t _msize(void* memblock);  // Windows平台下专用函数,非C语言标准函数
  • 返回malloc() & calloc() & realloc()等申请内存块的大小,参数是分配内存块的首地址,也就是malloc() & calloc() & realloc()等的返回值
int* p = (int*)malloc(sizeof(int));
int pSize = _msize(p);  // pSize == 4;
int* q = (int*)realloc(p, sizeof(int) * 2);
int qSize = _msize(q);  // qSize == 8; p和q可能相等也可能不相等

2. free

  • 用malloc()申请一块内存空间,OS会有一张表记录所申请空间的首地址和这块地址的长度
  • free(空间首地址),free会从表中查找到这块首地址对应的内存大小,一并释放掉
int* p = (int*)malloc(4);
free(p);
p = NULL; // 释放后要置为空指针
int* q = (int*)malloc(3);
free(q); // 会报错,int型指针一次操作 4Byte,这里只申请了 3Byte 相当去别人的地盘上拆东西,那肯定是不允许的
int* n = (int*)malloc(7); // 允许多申请,但是 int型指针一次只能操作 4Byte 多余的空间浪费了
free(n); // 释放时,从OS维护的表中查找到空间长度,会一并释放掉

注意

  1. free()不能去释放栈区的空间,栈区空间是由OS管理的,由OS进行申请和释放
  2. 释放空间后,指针需要置空,避免成为野指针
int* p = (int*)malloc(sizeof(int));
if (p == NULL) { // p 是空指针// 空间申请失败的错误处理
} else {// 申请成功,假设 p == 0X00000191D34DDAB0;free(p); // p == 0X00000191D34DDAB0; p有值,但是指向的内存空间已经被释放掉了,p就成了一个野指针了p = NULL; // 释放空间后,指针需要置空,避免成为野指针
}
int *p; //这种,定义完指针未初始化,也是野指针

下一篇:C/C++内存申请和释放(二)

C/C++内存申请和释放(一)相关推荐

  1. fork练习、从进程角度考虑堆区内存申请与释放的有关问题

    1.fork练习 1.1代码1; int main( int argc, char* argv[], char* envp[]) {int i = 0;for( ; i < 2; i++ ){f ...

  2. C/C++动态内存申请与释放

    20.1 理解指针的两种"改变" 普通变量(非指针,简单类 型变量)只能改变值:   1) int a = 100; 2) ... 3) a = 200;   第 1 行代码,声明 ...

  3. sk_buff整理笔记(三、内存申请和释放)

    承接上一篇blog--sk_buff整理笔记(二.操作函数),这篇是要来讲解下sk_buff结构的内存申请和释放函数.因为sk_buff结构是比较复杂的(并不是其本身结构复杂,而是其所指的数据区以及分 ...

  4. malloc的内存申请和释放

    一.malloc malloc是个库函数,使用时要包含<stdlib.h>这个头文件  malloc向内存申请空间时需要我们指定所需内存的大小,并且申请成功时,返回指向所申请的内存空间的指 ...

  5. 内存申请与释放(转)

    释放内存?那要看你怎么申请的了 new->delete;malloc->free;GlobalAlloc->GlobalFree;VirtualAlloc(Ex)->Virtu ...

  6. 结构体变量内存申请与释放

    目录 1.前言 2.常见结构类型 3.Demo 4.结束 1.前言 结构体是C.C++开发中不可或缺的数据结构,往往涉及到函数的入参以及出参等,也必然涉及到参数的初始化.对于字符串往往是需要在堆上开辟 ...

  7. C++之内存管理:申请与释放

    目录 前言 1.C/C++内存分布 1.1虚拟内存分段 1.2理解一些概念 1.2.1栈帧向下增长 1.2.2堆向上生长 1.2.3栈和堆会碰撞吗? 1.2.4关于const的说明 2.C语言中动态内 ...

  8. linux 用户进程结束后 malloc申请的内存会自动释放吗,进程退出后malloc的内存是否会被释放?

    当一个进程退出后,不管是正常退出或者是异常退出,操作系统都会释放这个进程的资源.包括这个进程分配的内存,打开的文件等等. 内存泄露的前提是进程一直在运行:进程一旦退出,所占的整个虚拟内存都被销毁,所有 ...

  9. 内存资源的申请与释放(CC++)

    在嵌入式系统中动态内存申请存在比一般系统编程时更严格的要求,这是因为嵌入式系统的内存空间往往是十分有限的,不经意的内存泄露会很快导致系统的崩溃. 所以一定要保证你的malloc和free成对出现,如果 ...

最新文章

  1. MySQL Workbench 怎么创建数据库
  2. wxWidgets:wxScrollBar类用法
  3. python字符串的美化_Python构造自定义方法来美化字典结构输出
  4. P3527-[POI2011]MET-Meteors【整体二分,树状数组】
  5. 用户文件夹安全权限设置只能复制即上传不能删除和修改名称和修改数据或写入数据等
  6. 理解 JS 回调函数中的 this
  7. Python-flask中数据库连接池DBUtils
  8. 使用python爬取网站源代码
  9. “清华同方同传”By软件:同方易教管理平台 V2.4
  10. 智慧交通:数智化地铁大屏管控运维平台
  11. 烟雨江南暂排第一,第四届橙瓜网络文学奖入围20年十佳奇幻大神
  12. 导数定义考法一网打尽
  13. 分析virtio-blk+qemu+spdk环境中virtio-blk不用执行virtqueue_kick操作通知后端处理IO的原因
  14. 微信Android热补丁实践演进之路-andFix / ClassLoader / Tinker
  15. 软件测试需要学习哪些技能?
  16. WIN32API串口接收数据简单测试
  17. 图标(Icon)和图标按钮(IconButton)
  18. C盘系统文档迁移工具,一键解决重装,C盘爆满
  19. 深度学习 - 其本质是什么?
  20. svg中path图形自适应_制作自己的自适应SVG图形和图表

热门文章

  1. Valve(维尔福软件公司) Half Life(半条命) CS(反恐精英)
  2. 二进制、八进制、十进制、十六进制的转换
  3. 蔡徐坤打篮球and源码
  4. GEE导出图像到本地结果全部为空
  5. 2022-08-04 Brighthouse: An Analytic DataWarehouse for Ad-hoc Queries
  6. open judge 1.5 18:鸡尾酒疗法
  7. Matlab入门--画一个三维山顶图(将来会画马鞍面)
  8. 美赛数据(各国统计数据网站大全)
  9. python与pdf与word(datawhale组队学习task3)
  10. 推荐阅读:太极拳的奥妙-专访七十肖维佳老翁现场展示