C语言之动态内存管理(malloc calloc realloc)

C语言之动态内存管理

文章目录

  • C语言之动态内存管理
    • 1. 为什么要有动态内存管理
    • 2. malloc 和 free
      • 2.1 malloc
      • 2.2 free
      • 2.3 例子
    • 3. calloc 和 realloc
      • 3.1 calloc
      • 3.2 realloc
    • 4. 常见的动态内存错误
      • 4.1 对NULL指针的解引⽤操作
      • 4.2 对动态开辟空间的越界访问
      • 4.3 对⾮动态开辟内存使⽤free释放
      • 4.4 使⽤free释放⼀块动态开辟内存的⼀部分
      • 4.5 对同⼀块动态内存多次释放
      • 4.6 动态开辟内存忘记释放(内存泄漏)
    • 5. 总结

1. 为什么要有动态内存管理

我们已经掌握的内存开辟⽅式有:

#include <stdio.h>int main()
{int val = 20;int arr[10] = { 0 };return 0;
}

上述的开辟空间的⽅式有两个特点
• 空间开辟大小是固定的

•数组在申明的时候,必须指定数组的⻓度,数组空间⼀旦确定了⼤⼩不能调整

所以C语⾔引⼊了动态内存开辟,让程序员⾃⼰可以申请和释放空间

2. malloc 和 free

mallocfree函数都是在stdlib.h头文件中声明的

2.1 malloc

C语言中提供了一个动态内存开辟的函数:

void* malloc (size_t size);

其中size为要开辟的内存空间的大小,单位为字节

这个函数向内存申请⼀块连续可⽤的空间,并返回指向这块空间的指针

•如果开辟成功,则返回一个指向开辟好的内存空间的指针
•如果开辟失败,则返回一个NULL指针
•返回类型为void *,因为malloc函数不知道要开辟什么类型的内存空间,只知道要开辟的大小
•如果参数为0,malloc函数的行为标准是未定义的,取决于编译器

2.2 free

C语言还提供了一个的函数,专门用来做动态内存的释放和回收的:

void free (void* ptr);

ptr为要释放内存空间的指针

•如果参数ptr指向的内存空间不是动态开辟的,那么free函数的行为是未定义的
•如果参数ptrNULL,则函数什么都不做

如果不对malloc calloc realloc 开辟的空间进行释放,即使出了作用域也不会销毁,有可能导致内存泄漏

释放的方式
1. free
2. 直到程序结束,由操作系统释放

2.3 例子

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int* p = (int*)malloc(10 * sizeof(int)); //开辟40个字节的空间//判断是否为NULL指针if (p == NULL){perror("malloc fail\n"); //perror为错误信息打印return 1;}//使用int i = 0;for (i = 0; i < 10; i++){*(p + i) = i;//*p = i;   //如果使用这种方法,p指针向后走了,在下面打印时,就找不到首元素的地址了//p++;}//打印for (i = 0; i < 10; i++){printf("%d ", *(p + i));}free(p);  //释放p = NULL; //将指针置NULL,如果不置NULL,下面解引用p时,p就是野指针return 0;
}

代码运行结果:>
0 1 2 3 4 5 6 7 8 9

malloc开辟空间时,是不会给空间初始化的,如果直接打印,会打印出随机值

3. calloc 和 realloc

3.1 calloc

C语⾔还提供了⼀个函数叫 calloccalloc 函数也⽤来动态内存分配

void* calloc (size_t num, size_t size);

num为要开辟的元素个数
size为开辟元素的元素大小,单位为字节

calloc为开辟num个大小为size元素的内存空间,并且将内存中每个字节初始化为0
calloc的使用方法和malloc一样,主要区别在于calloc会初始化元素

例子:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int* p = (int*)calloc(10 ,sizeof(int));//判断是否为NULL指针if (p == NULL){perror("calloc fail\n");return 1;}int i = 0;for (i = 0; i < 10; i++){printf("%d ", *(p + i));}return 0;
}

代码运行结果:>
0 0 0 0 0 0 0 0 0 0

3.2 realloc

C语言中有一个函数用来调整动态内存开辟后的大小

void* realloc (void* ptr, size_t size);

ptr为要调整的内存地址
size为调整后的内存大小

realloc函数的出现让动态内存管理更加灵活
• 当我们发现我们使用malloc calloc realloc申请的内存空间不够时,我们可以使用realloc进行扩容

•返回值为调整之后的内存的起始位置(不一定是原内存地址)
•如果开辟失败则返回一个NULL
•如果开辟成功则分以下两个情况:

情况1:原有空间之后有⾜够⼤的空间
情况2:原有空间之后没有⾜够⼤的空间

在这里插入图片描述

情况1:在原有内存后边直接追加空间,原来的空间的数据不变
情况2:原有内存之后的空间不足以最加空间,那么realloc会在堆区中找到一块足够开辟新大小的空间,将旧空间中的数据拷贝到新空间,并且将旧空间释放,同时返回新空间起始位置的地址

realloc的用法除了为开辟的内存进行扩容,也可以和malloc一样

例子:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int* p = (int*)malloc(10 * sizeof(int));//判断是否为NULL指针if (p == NULL){perror("malloc fail\n");return 1;}int* tmp = (int*)realloc(p, 100*sizeof(int));if (tmp != NULL){p = tmp;}else{perror("relloc fail\n");return 1;}//使用//..........free(p);p = NULL;return 0;
}

在这里插入图片描述
这次的运行结果就是情况2(开辟100个字节的大小时,可能会出现),当后面的空间不够时, realloc就会找一块新的空间
在这里插入图片描述
这次只开辟了40个字节的空间,属于情况1,后面的空间足够时, realloc会直接在后面追加空间

4. 常见的动态内存错误

4.1 对NULL指针的解引⽤操作

#include <stdio.h>
#include <stdlib.h>int main()
{int* p = (int*)malloc(40);*p = 20; //如果malloc开辟空间失败,p可能是NULL,此时p为野指针return 0;
}

在VS2022中,编译器会进行提示,我们得对可能出现NULL的情况进行处理
在这里插入图片描述

在使用malloc calloc realloc开辟空间时,最好对返回值进行判断,当不为NULL再使用

4.2 对动态开辟空间的越界访问

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int* p = (int*)malloc(40);if (p == NULL){perror("malloc fail\n");return 1;}int i = 0;for (i = 0; i <= 10; i++)  //只有10个元素的空间,却访问了第11个元素,访问越界了{*(p + i) = i;}free(p);p = NULL;return 0;
}

在这里插入图片描述

4.3 对⾮动态开辟内存使⽤free释放

#include <stdio.h>
#include <stdlib.h>int main()
{int a = 10;int* p = &a;free(p);return 0;
}

在这里插入图片描述

当用free释放了不是由malloc calloc realloc开辟的空间时,就会报错

4.4 使⽤free释放⼀块动态开辟内存的⼀部分

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int a = 10;int* p = (int*)malloc(40);if (p == NULL){perror("malloc fail\n");return 1;}//使用//......p++;free(p);p = NULL;return 0;
}

在这里插入图片描述
当用free释放了开辟空间的一部分时,就会报错

4.5 对同⼀块动态内存多次释放

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int a = 10;int* p = (int*)malloc(40);if (p == NULL){perror("malloc fail\n");return 1;}//使用//......free(p);free(p);p = NULL;return 0;
}

在这里插入图片描述
对一块动态开辟的内存进行多次free释放

在上述代码中如果free释放NULL,则没有问题,因为free的参数为NULL时,则什么都不做

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int a = 10;int* p = (int*)malloc(40);if (p == NULL){perror("malloc fail\n");return 1;}//使用//......free(p);p = NULL;free(p);p = NULL;return 0;
}

4.6 动态开辟内存忘记释放(内存泄漏)

#include <stdio.h>
#include <stdlib.h>void test()
{int* p = (int*)malloc(100);if (p != NULL){*p = 20;}
}
int main()
{test();while (1); //死循环,让程序不结束return 0;
}

当动态开辟的内存不释放时,就会一存在,在上述代码中,调用了test函数,开辟了100个字节的空间,同时赋值,出函数时,p被销毁了,但是开辟的空间并没有被销毁,没人可以使用,也没人可以释放,就会导致内存泄漏

5. 总结

一丶
在使用malloc calloc realloc开辟的空间时,要对其进行判断,当不为NULL的再进行使用
二丶
当不使用动态开辟的内存时,将其free释放,同时将指针置NULL,防止可能出现的内存泄露和野指针
三丶
不对不是动态开辟的空间free,不连续对动态开辟的空间free,同时free动态开辟的空间时,要给开辟的起始地址,不能free部分空间

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.hqwc.cn/news/257890.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

华三AC+AP部署无线基本配置

拓扑图&#xff1a; 1、创建管理VLAN与AP管理VLAN、终端接入VLAN、配置管理VLAN IP地址 设备管理VLAN 56、AP管理VLAN 101、终端接入VLAN 10 AC、HeXin、JieRu。&#xff08;创建VLAN&#xff09;[H3C]vlan 101[H3C-vlan101]description AP-vlan[H3C]vlan 56[H3C-vlan56]desc…

OpenCL学习笔记(一)开发环境搭建(win10+vs2019)

前言 异构编程开发&#xff0c;在高性能编程中有重要的&#xff0c;笔者本次只简单介绍下&#xff0c;如何搭建简单的开发环境&#xff0c;可以供有需要的小伙伴们开发测试使用 一、获取opencl的sdk库 1.使用cuda库 若本机有Nvidia的显卡&#xff0c;在安装cuda库后&#x…

【洛谷】更换头像

错误展示 今天换头像的时候发现一直换不了&#xff0c;即使显示修改成功&#xff0c;然后我等了半个多小时也还没换好 解决办法 上传成功头像后&#xff0c;按ctrl F5 结果 更新成功&#xff01;

题目:分糖果(蓝桥OJ 2928)

题目描述&#xff1a; 解题思路&#xff1a; 本题采用贪心思想 图解 题解&#xff1a; #include<bits/stdc.h> using namespace std;const int N 1e6 9; char s[N];//写字符串数组的一种方法,像数组一样***int main() {int n, x;cin >> n >> x;for(int …

node14升级node16之后无法启动处理

node从14升级到16之后&#xff0c;项目就无法启动了&#xff0c;研究了webpack3升级5&#xff0c;研究好几个小时都无法启动&#xff0c;最后发现&#xff0c;微微升级几个版本就可以了。webpack还是3 新版的依赖 "scripts": {"dev": "webpack-dev-se…

Salesforce×阿里云,影响几何?

实际上&#xff0c;从这个视角来看&#xff0c;Salesforce和阿里云的合作也恰在成为着这个市场的一个新催化剂。“期待Salesforce能给中国市场带来一些新的增量&#xff0c;包括对合作伙伴的态度&#xff0c;对产品的态度等等。”一位CRM相关人士告诉我们。 那么&#xff0c;阿…

外贸CRM系统的含义,外贸CRM有什么作用?

外贸CRM管理系统简单方便&#xff0c;配备多种功能&#xff0c;受到从业者的青睐。可以帮助外贸公司管理客户信息&#xff0c;进行准确营销&#xff0c;随后获得更多销售额。这篇文章向您介绍&#xff1a;外贸CRM是什么意思&#xff1f;以及外贸CRM系统的意义 1.外贸CRM是什么意…

销售管道是什么?CRM销售管道有什么作用?

在之前的文章里&#xff0c;我曾经提到过CRM系统中的销售管道。销售管道可以理解为您的销售过程管道&#xff0c;针对不同的产品线&#xff0c;或不同的销售方式&#xff0c;来创建不同的管道流程。今天我们就来说说&#xff0c;CRM销售管道有什么作用&#xff1f; 销售管道重要…

记录 | windows网络诊断报错:无法自动检测此网络的代理设置

windows网络诊断报错&#xff1a;无法自动检测此网络的代理设置 【解决办法1】 打开Microsoft Edge&#xff0c;找到设置&#xff0c;点击高级&#xff0c;打开代理设置&#xff0c;把所有的都关了&#xff0c;就好了。 【解决办法2】 开始 > 运行 > 输入"cmd"…

Ubuntu18安装(重启黑屏问题)

1. F10 进入bios&#xff0c;选择u盘里的ubuntu镜像 2.进入使用ubuntu&#xff0c;下载 3.重启&#xff0c;esc 4.ubuntu 安e进入 5. nomodeset&#xff08;&#xff09; F10 保存启动 6. 7.没有网 手机usb提供网络 下载有限网卡驱动

vue实现可拖拽列表

直接上代码 <!-- vue实现可拖拽列表 --> <template><div><button click"logcolig">打印数据</button><TransitionGroup name"list" tag"div" class"container"><divclass"item"v-f…

机器学习与低代码开发:创新驱动的双剑合璧

引言 随着科技的日新月异&#xff0c;机器学习和低代码开发已经成为引领技术行业变革的两大重要趋势。机器学习通过模拟人类的学习方式&#xff0c;让计算机具备了自我学习和预测的能力&#xff0c;打破了传统计算机程序的局限性。而低代码开发则以简化软件开发过程为目标&…