【C语言】------ 动态内存分配

article/2025/9/14 21:46:39

动态内存开辟详解

  • 动态内存分配
    • 什么是动态内存分配?
  • 一、为什么使用动态内存分配呢?
  • 二、动态内存函数
    • 1.malloc和free
    • 2.calloc和realloc
  • 三、常见的动态内存错误
    • 1.对`NULL指针`的解引用操作
    • 2.对动态内存开辟的`越界访问`
    • 3.向free传递一个非malloc函数返回的指针
    • 4.动态开辟内存忘记释放造成`内存泄漏`
  • 总结


动态内存分配

什么是动态内存分配?

所谓的动态内存分配,就是在程序执行的过程中动态地分配或者回收存储空间的分配内存的方法。动态内存分配不像数组等静态内存分配方法那样需要预先分配存储空间,而是由系统根据程序的需求即时分配


一、为什么使用动态内存分配呢?

示例:当我们在声明一个数组的时候,我们需要指定数组的长度,它所需要的内存编译时分配。但是在程序运行的时候我们才可能知道真实所需要的长度,因为它所需要的内存空间取决于输入的数据

char arr[10] = {0};//在栈空间上开辟10个字节的连续空间

这种方法的优点是很简单,但是如果程序需要使用的元素数量超过了声明的长度,就无法处理这种情况。那怎么办呢?我们很容易就想到那就把数组声明的大一点,这确实是一个办法,然而这也会使得很大部分的内存空间浪费掉了,所以我们引出动态内存分配

我们先看看内存的分区
在这里插入图片描述

二、动态内存函数

1.malloc和free

C语言函数库提供了俩个函数,mallocfree,分别用来执行动态内存的分配和释放,当一个程序需要内存时,就会调用malloc函数,malloc从内存池()中提取一块内存,并向该程序返回一个指向这块内存的指针(此块内存此时并未初始化)。

void* malloc (size_t size);
void free(void* ptr);

1.它们的头文件在stdlib.h中声明,malloc的参数是需要分配的内存字节数,若开辟成功,则返回一个指向被开辟的内存块的起始位置的指针,注意malloc所分配的是一块连续的内存;如果内存池是空的或者它的可用内存无法满足你的请求的时候,malloc则返回一个NULL空指针,所以对每个malloc返回的指针都应该进行检查。
由于返回值的类型是void*,所以mall函数并不知道开辟空间的类型是整型、浮点型、结构或者是数组,具体使用的时候可以转换为其他任何类型的指针。
2.free函数专门用来做动态内存的释放和回收的:

如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。
如果参数 ptr 是NULL指针,则函数什么事都不做。

我们看下面这个例子:

int main()
{//int arr[10] = { 0 };//动态内存开辟int* p = (int*)malloc(40);if (p == NULL)//判断p是否为空指针{printf("%s\n", strerror(errno));return 1;//异常返回}//使用内存int i = 0;for (i = 0; i < 10; i++){*(p + i) = i;printf("%d ", *(p + i));}//没有free//当程序退出的时候,系统自动回收内存空间free(p);p = NULL;return 0;//正常返回
}

判断p不为空指针是进行(使用代码)
在这里插入图片描述
将申请的空间内存释放回收:

free( p);
p = NULL;

在这里插入图片描述

2.calloc和realloc

(1)calloc函数也用于内存分配,它与malloc之间的主要区别在于calloc返回地址之前把申请的空间的每个字节初始化为全0。另外一个小的区别在于它们请求内存数量的方式不同。calloc的参数包括所需要元素的数量和每个元素的字节数,根据这些值计算出需要开辟的内存

void* calloc (size_t num, size_t size);

例如:

int main()
{int* p = (int*)calloc(10, sizeof(int));if (p == NULL){printf("%s\n", strerror(errno));return 1;}//打印int i = 0;for (i = 0; i < 10; i++){printf("%d ", *(p + i));}//释放free(p);p = NULL;return 0;
}

我们看这段代码运行的结果:
在这里插入图片描述
我们什么也没有做,但是代码打印的值全被赋成了0,说明calloc函数在开辟好空间之后,把内容初始化成0,然后把起始地址返回来。
所以在calloc和malloc之间如何选择我们就有了一点的了解,如果你需要初始化则选择calloc。
(2)realloc函数用于修饰一个原先已经分配的内存块的大小,它可以使一块内存块扩大或者缩小:
realloc函数原型:

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

  • ptr是要调整的内存地址
  • size是调整后的新的大小

1.当用于扩大一个内存块时,这块内存原先的内容将依然保留,新增加的内存添加到原先内存块的后面,且新内存并没有进行初始化。

2.当缩小一块内存的时候,该内存块尾部的部分内存被拿掉,剩下的原先内容保留不变。
3. 当原先的内存块无法改变大小,realloc将分配另一块正确大小的内存,并把原先内存的内容复制到新的内存块上。所以在使用realloc之后,不能再使用指向旧的内存的指针,而是该用realloc所返回的新指针。

在这里插入图片描述

举个例子:

int main()
{int* p = (int*)malloc(40);if (NULL == p){printf("%s\n", strerror(errno));return 1;}//使用  1 2 3 4 5 6 7 8 9 10int i = 0;for (i = 0; i < 10; i++){*(p + i) = i + 1;}//扩容int* ptr = (int*)realloc(p, 80); //用ptr来接收if (ptr != NULL){p = ptr;//p得到有效的地址(/新)}//使用for (i = 0; i < 10; i++){printf("%d ", *(p + i));}//释放 free(p);p = NULL;return 0;

在这里插入图片描述

三、常见的动态内存错误

1.对NULL指针的解引用操作

不检查从malloc函数返回的指针是否为空

void test()
{int *p = (int *)malloc(20);*p = 20;//如果p的值是NULL,会出错free(p);
}

2.对动态内存开辟的越界访问

访问动态分配的内存之外的区域

void test()
{int i = 0;int *p = (int *)malloc(10*sizeof(int));if(NULL == p){printf("%s\n", strerror(errno));return 1;}for(i=0; i<=10; i++){*(p+i) = i;//当i是10的时候越界访问}free(p);p=NULL:return 0;
}

3.向free传递一个非malloc函数返回的指针

对非动态内存开辟使用free释放

void test()
{int a = 10;int *p = &a;free(p);//ok?
}

4.动态开辟内存忘记释放造成内存泄漏

void TEST()
{int *p = (int *)malloc(100);if(NULL != p){*p = 20;}
}
int main()
{TEST();return 0}

内存泄漏会增加程序的体积,可能导致程序或系统崩溃,所以

一定要正确的释放内存


总结

1.malloc和calloc函数用于动态分配一块内存,并返回一个指向该内存的指针,malloc的参数就是需要分配的内存的字节数,而calloc的参数时需要分配的元素个数和每个元素的长度;calloc函数在返回时将内存初始化为0,而malloc则不会进行初始化。

2.realloc函数可以调整动态内存分配的大小。以及出现的两种情况

3.如果请求内存开辟失败,malloc、calloc和realloc函数返回的是一个NULL指针。若一个指针不是从这些函数返回的,则它不能作为参数传递给free函数,同时,我们也不能只释放一块内存的一部分。

到这里,相信大家对动态内存分配有了一定的了解与掌握吧!我在这里是把我所学习到的和书上的知识内容做个总结与叙述,希望能有所帮助,谢谢。


http://chatgpt.dhexx.cn/article/a34ghBno.shtml

相关文章

【C语言】为什么存在动态内存分配

文章目录 前言一、动态内存分配定义 二、动态内存分配的意义1.可以控制所开辟的内存大小2.可以多次利用这部分空间 三&#xff0c;动态内存函数的介绍 前言 提示&#xff1a;我们先来看一个在vs编译器下&#xff0c;同学们常犯的错误 注意&#xff1a;在VS编译器下C语言是不支…

静态内存分配与动态内存分配

静态内存分配与动态内存分配 动机 平时看c/c的书籍时&#xff0c;总会看到一种观点&#xff0c;说是C/C语言使用的时候动态内存分配是最重要的&#xff0c;使用malloc等函数分配的内存必须要释放&#xff0c;否则及其容易出现内存泄露。但是自己有时候挺奇怪的&#xff0c;啥…

JVM内存分配机制

Java虚拟机最重要的工作就是如何给对象分配内存空间&#xff0c;以及通过GC如何回收已经不再使用的内存空间。这篇文章主要介绍JVM中的Java对象是创建过程、对象内存的分配机制以及对象内存的回收机制。 一、对象的创建 在前面的文章《JVM类加载机制》中讲过&#xff0c;JVM中…

C语言动态内存分配详解

文章目录 前言一、为什么存在动态内存分配1、已掌握的内存开辟方式2、上述开辟空间方式的特点3、为什么存在动态内存分配 二、动态内存函数的介绍1、malloc2、free3、calloc4、realloc 三、常见的动态内存错误1、对NULL指针的解引用操作2、对动态开辟内存的越界访问3、对非动态…

动态内存管理(内存的分配与回收)详解

** 1. 数据结构之动态内存管理机制 ** 通过前面的学习&#xff0c;介绍很多具体的数据结构的存储以及遍历的方式&#xff0c;过程中只是很表面地介绍了数据的存储&#xff0c;而没有涉及到更底层的有关的存储空间的分配与回收&#xff0c;从本节开始将做更深入地介绍。 在使…

关于C++的动态内存分配

为什么要使用动态内存分配&#xff1f; 比较常见的情况是当我们使用一个数组时&#xff0c;我们需要去声明它&#xff0c;同时我们还需要提供给它一个编译时常量用于指定数组的长度。但是&#xff0c;我们有时候需要的数组并不是定长的。例如&#xff0c;我们要存储一个班级所…

【C语言】动态内存的分配

目录 &#x1f34b;&#x1f34b;前言 &#x1f34b;&#x1f34b;动态内存分配的定义 &#x1f34b;&#x1f34b;动态内存的优势 &#x1f34b;&#x1f34b;<1> 可以控制内存的大小 &#x1f34b;&#x1f34b;<2> 可以多次利用这部分空间 &#x1f34b;…

动态内存分配函数

一、静态存储分配与动态存储分配&#xff1a; &#xff08;1&#xff09;静态存储分配 通常定义变量&#xff08;或对象&#xff09;&#xff0c;编译器在编译时都可以根据该变量&#xff08;或对象&#xff09;的类型知道所需内存空间的大小&#xff0c;从而系统在适当的时候…

内存动态分区分配算法

文章目录 动态分区分配1. 首次适应算法&#xff08;First Fit&#xff09;2. 邻近适应算法&#xff08;Next Fit&#xff09;3. 最佳适应算法&#xff08;Best Fit&#xff09;4. 最坏适应算法&#xff08;Next Fit&#xff09; 总结 动态分区分配 所谓动态分区分配&#xff0…

C++之new动态分配内存生成数组

【任务1】&#xff1a; 编写并测试3*3矩阵转置函数&#xff0c;使用数组保存3*3矩阵。 实验要求 &#xff08;1&#xff09; 转置函数参数为二维数组&#xff1b; &#xff08;2&#xff09; 在main函数中实现输入、输出 #include <iostream> using namespace std…

C语言中的动态内存分配

大家好&#xff0c;今天简单讲一讲C语言中的动态内存分配。 补充&#xff1a;C程序中的内存块。 在C程序中&#xff0c;通常将内存划分为以下六个区域&#xff1a; &#xff08;1&#xff09;内核区域。这块区域是操作系统的&#xff0c;用户不能使用。 &#xff08;2&…

C语言动态分配内存

C语言动态分配内存 malloc 动态开辟内存的函数&#xff1a; void* malloc (size_t size); 这个函数向内存申请⼀块连续可⽤的空间&#xff0c;并返回指向这块空间的指针 如果开辟成功&#xff0c;则返回⼀个指向开辟好空间的指针 如果开辟失败&#xff0c;则返回⼀个NULL…

C/C++动态分配内存的几种方法

使用C/C编程时&#xff0c;会经常动态分配内存&#xff0c;以便合理使用内存&#xff0c;本文主要讲述动态内存分配的几种方法及一些原理&#xff0c;理解不深刻之处欢迎指教。 引言 为什么要进行动态内存分配&#xff1f;以数组为例&#xff0c;数组元素在内存中存储的地址是…

sadad

出版社&#xff1a; 河北少年儿童出版社 ISBN&#xff1a;12176804 版次&#xff1a;1 商品编码&#xff1a;12176804 包装&#xff1a;精装 丛书名&#xff1a; 幼儿家庭课堂 开本&#xff1a;12开 出版时间&#xff1a;2017-05-01 用纸&#xff1a;铜版纸 适读人群 &#xff…

SAADC

电流 :EasyDMA draws about 1.2 mA, and added to the 700 uASAADC current it sums to ~2 mA。和实测差不多 官方例程解析:TIMER + PPI +SAADC。 特征: 1、 支持8/10/12位分辨率,和14位过采样分辨率 2、 8通道:单端输入最多可同时配置8个通道 和 差分输入…

SAZS~

本文提出了一种新颖的形状感知零样本语义分割&#xff08;SAZS&#xff09;框架&#xff0c;利用大规模预训练视觉语言模型特征空间中丰富的先验知识&#xff0c;并通过在边界检测约束任务上进行联合训练来融合形状感知 论文链接&#xff1a;https://arxiv.org/abs/2304.08491…

SASSD

Structure Aware Single-stage 3D Object Detection from Point Cloud 文章&#xff1a;SA-SSD 代码&#xff1a;SA-SSD 这是来自香港理工大学和达摩院的一篇文章&#xff0c;一作是为在读的博士生&#xff0c;何晨航。通讯作者是张磊&#xff0c;香港理工大学的博导&#x…

Sarsa

基本概念 Sarsa Sarsa算法与Q-Learning算法极为相似&#xff0c;‘sarsa’五个字母的含义即s(当前状态)&#xff0c;a(当前行为)&#xff0c;r(奖励)&#xff0c;s(下一步状态)&#xff0c;a(下一步行为)&#xff0c;也就是说我们在进行这一步操作时已经想到当前s对应的a&…

反激电源

扶苗平衡&#xff1a;是针对电感的&#xff0c;不是针对变压器的&#xff1b;是电感稳态期间的情况&#xff0c;不是暂态期间的情况&#xff1b; 一个周期内&#xff0c;电感开通期间两端的电压乘以开通时间电感关断期间两端电压乘以关断时间。 对于反激变压器&#xff0c;其实…

Part-Ⅰ2. 稳态变换器分析原则(二)

2.4 Cuk电路分析 作为第二个示例&#xff0c;变换器可以实现类似buck-boost电路功能&#xff1a;升高或降低电压幅值&#xff0c;并且转换极性。其电路如如下。 该转换器通过电容进行能量传输。当开关打到位置2时&#xff0c;电容 C 1 C_1 C1​通过 L 1 L_1 L1​连到输入电源…