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

article/2025/9/14 22:42:49

目录

🍋🍋前言

🍋🍋动态内存分配的定义

🍋🍋动态内存的优势

🍋🍋<1>  可以控制内存的大小

🍋🍋<2> 可以多次利用这部分空间

🍋🍋<3>不占用栈区的内存

🍋🍋malloc calloc realloc和free函数的介绍

🍋🍋​​​动态空间的申请与释放

         🍋🍋为单个对象分配空间

🍋🍋 为数组分配空间

🍋🍋改变申请的动态内存(realloc的使用)

🍋🍋扩容可能有三种情况


👀前言

给数组分配多大的空间?

你是否和初学C时的我一样,有过这样的疑问。

这一期博客就来聊一聊动态内存的分配

读完这篇文章,你可能对内存的分配有一个更好的理解

📢动态内存分配的定义

首先我们要搞清楚什么是动态内存的分配

平常我们定义的数组,都是在栈区分配的空间,都是分配的空间都是固定的大小

这种分配固定大小的内存分配方法称之为静态内存分配

与静态内存相对的,就是可以控制内存的分配的动态内存分配

📢注意:这里动态内存分配的空间是在堆区申请的,不是在栈区申请的

📌这里要讲一下什么是栈区,什么是堆区

内存的空间并不是都是一样的,在学习C语言时,提到的区域大致上分为栈区,堆区,和静态区。就比如说在一个车间一样,不同的区域做着不同的事,就有不同的功能,但是这些不同的功能又不是毫不相关的,他们彼此联系,相互构成整个内存空间.

📢动态内存的优势

<1>  可以控制内存的大小

在很多时候,我们申请的空间是未知的

就比如说通讯录,在刚刚开始用的时候很小的空间就足够了,但是在未来你不知道你需要存下多少个号码,这时候就存在一个问题,你定的空间需要多少个字节,当申请的太少,就会出现存不下去的情况,如果存的空间过大,有会造成一定的浪费。 

在动态内存分配就可以避免这个问题,你可以运用 reallac 控制大小,当内存达到申请的空间时,就会主动扩容,也就是再次向内存申请空间。

<2> 可以多次利用这部分空间

静态内存分配利用的空间,整个程序结束才会释放给系统

而动态内存分配的空间,只能在函数运行结束后由系统自动释放,需要用户主动去释放,可以通过利用完(就比如说打印元素,打印完),用户再通过 free函数释放 这块申请的空间,当再次用动态内存申请空间时,就可以再次利用这块空间,这样也能在一定程度上,可以节省一定的空间。

<3>不占用栈区的内存

假设栈区定义了变量

而每个变量分配内存时,之间又有一定的间隙

当定义的变量足够多时,空隙也会很多

这时候向系统申请一个比较大且连续的空间时,虽然有足够的空间,但是缺少了连续的空间

就无法申请到这部分空间

所以动态内存在堆区申请,就完全不必担心栈区的空间不够的问题


说到这里,你是不是有一个疑惑,为什么空间的内存存在栈区和堆区之分 

如果感兴趣,可以参考这个回答——为什么存在栈区堆区

📢malloc calloc realloc和free函数的介绍

在动态内存的分配中,离不开malloc与calloc,这两个函数都是向内存申请空间

                          calloc
头文件               #include <stdlib.h> 
格式                   void *calloc(size_t num, size_t size);
功能                   为num个大小为size字节的对象分配存储空间,该空间内的所有位都会初始化为。
返回值                若分配成功,则返回一个指向已分配的空间开头的指针;若分配失败,则返回空指针
                          malloc
头文件               #include <stdlib.h>
格式                    void *malloc(size_t size);
功能                    为大小为size字节的对象分配存储空间,此存储空间中的初始值不确定
返回值                 若分配成功,则返回一个指向已分配的空间开头的指针;若分配失败,则返回空指针

这两个函数都是向系统申请动态内存空间,他们的头文件,返回值和功能大致都是相同的

不同的是calloc函数开辟的空间,就会将空间的内容全部初始话为零

而,malloc函数向系统申请的空间,空间的值都是随机的

                realloc

头文件     #include <stdlib.h>
格式         void *realloc(void *mem_address, unsigned int newsize);
功能         先判断当前的指针是否有足够的连续空间,如果有,扩大mem_address指向的地址,并且将                            mem_address返回,如果空间不够,先按照newsize指定的大小分配空间,将原有数据从头到尾拷贝                  到新分配的内存区域,而后释放原来mem_address所指内存区域(注意:原来指针是自动释放,不                    需要使用free),同时返回新分配的内存区域的首地址。即重新分配存储器块的地址。
返回值      如果重新分配成功则返回指向被分配内存的指针,否则返回空指针NULL。
                         free
头文件             #include <stdlib.h>
格式                 void free(void *ptr);
功能                 释放ptr指向的空间,让这部分空间能继续用于之后的动态分配。当ptr为空指针时,不执行任何                          操作。除此之外,当实际参数与之前通过calloc函数、mal1oc函数或realloc函数返回的指针不                          一致时,或者ptr指向的空间已经通过调用 free或real1oc被释放时,则作未定义处理
返回值             无

📢动态空间的申请与释放

讲完动态内存申请的相关函数,那具体的代码实现是什么呢

<1>    double *x;

<2>    x=calloc(1,sizeof(double))或者x=malloc(sizeof(double));

<3>    free;

 下面动态分配的内存赋值并显示

📢为单个对象分配空间

#include<stdio.h>
#include<stdlib.h>
int main()//动态内存的赋值与显示
{int* a;a = malloc(sizeof(int));    //分配动态内存if (a == NULL)              //是否成功分配了储存空间,否则返回分配失败printf("分配失败");else{*a = 20;printf("*a=%d\n", *a);free(a);                //释放}return 0;
}

📢 为数组分配空间

#include<stdio.h>
#include<stdlib.h>
int main()//动态内存的赋值与显示
{int n = 0;  int* a; int i = 0;printf("输入分配空间元素的个数:>");scanf_s("%d", &n);a =(int *) calloc(n,sizeof(int));if (a == NULL)printf("分配失败");else{for (i = 0; i < n; i++){*(a + i) = i;printf("*a=%d\n", *(a+i));}free(a);}return 0;
}

这里其实没有“为数组开辟的空间”这一说

因为动态申请的空间都是一个一个的“

不难发现,calloc与malloc的差别并不大,只有第一个参数不同

在这两行代码中,存在着一个小细节 

a = malloc(sizeof(int));

a =(int *) calloc(n,sizeof(int));

这两者的差别不仅仅是函数的不同,其中后者有强制类型转换,而前者没有

实际上在C语言的标准上,有无强制类型转换都是行得通的(当然在c++必须将强制类型转换)

因为无论是calloc还是malloc,他们的返回值都是void* ,这里的void*实际上可以转换为int*类型或者其他类型,换句话说,就是返回的指针是兼容所有类型的万能指针。

📢即指向void型的指针可以指向任意类型的对象,是一种特殊类型的指针。

指向void型的指针的值可以赋给指向任意类型的指针,反之亦可。

📢改变申请的动态内存(realloc的使用)

#include<stdlib.h>
#include<stdio.h>
int main()
{int* a; int i = 0;a=(int *)calloc(10,sizeof(int));if (a == NULL)printf("分配失败");//使用else{for (i = 0; i < 10; i++){*(a + i) = i;printf("*a=%d\n", *(a + i));}//需要扩容int* ret = realloc(a, 80);if (ret != NULL){a = ret;}free(a);a=NULL;}
}

📢扩容不能直接就a=realloc(a, 80),需要中间引一个中间变量*ret

📢扩容可能有三种情况

情况一(在a的地址处,有空余的空间来扩容)

情况二 (在a的地址处,没有空余的空间来扩容,但是有其他的空间可存储扩容后的空间)

 情况三(reallo调整空间失败)

 在这三种情况中,第一种的地址不变

第二种会在一个新的地方申请足够大的地方,此时的地址不在是a原先的地址

第三种就扩容失败,就会导致扩容前申请的空间,也发生了改变,所以不能直接用a来重新赋值,


 欢迎点赞收藏加关注,如若有问题可以提出来😁😁😁😁 


http://chatgpt.dhexx.cn/article/1pxa5RKx.shtml

相关文章

动态内存分配函数

一、静态存储分配与动态存储分配&#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​连到输入电源…

buck-boost基本模型

1.电容模型 2.电感模型 3.伏秒平衡 4.电容电感 5.BUCK 6.boost 7.参考资料 《开关电源的基本概念和分析方法》

2、Principles of Steady-State Converter Analysis

2、Principles of Steady-State Converter Analysis 伏秒平衡与安秒平衡 在电路平衡时&#xff0c;作为存储元件的电感与电容&#xff0c;其储存的能量不可能无穷大&#xff0c;必然会稳在一个稳定值。 稳定后必然有在一个开关周期内有 所以 在求解时&#xff0c;只要根据…

1dB压缩点和三阶交调点、相位差与延时

1dB压缩点与三阶交调点 要知道放大器是一个非线性系统&#xff0c;传输函数基本用泰勒级数表示 如果输入信号幅度很小&#xff0c;那么上式中2次及以上的项就可以忽略而成为小信号的情况。在许多情况下我们可以忽略3次以上的项。 如果输入一个正弦信号 1、可以看到一个单频率…

半桥电路注意事项

应注意的几点问题 偏磁问题 原因&#xff1a;由于两个电容连接点A的电位是随Q1、Q2导通情况而浮动的&#xff0c;所以能够自动的平衡每个晶体管开关的伏秒值&#xff0c;当浮动不满足要求时&#xff0c;假设 Q1、Q2具有不同的开关特性&#xff0c;即在相同的基极脉冲宽度tt1下&…

有关DC/DC和LDO 的详细介绍以及两者的对比----之DC/DC篇

目录 DC/DC工作原理 电感电压伏秒平衡定律 开关电源的三种基本拓扑 同步整流技术 DC/DC电源调制方式 DC/DC芯片的内部构造 DC/DC电路的硬件设计 总结一下DC/DC和LDO的原理&#xff0c;区别和应用&#xff0c;以下内容部分为自己原创&#xff0c;部分在网上找的资料并加上…

BUCK中的电感值---开关电源篇(4)

前一篇&#xff0c;BUCK电路&#xff0c;了解到功率电感中的电流是呈现为以开关频率为周期的伏秒平衡态。 即dI * L dV * dt 1&#xff09; 相同伏秒变化的情况下&#xff0c;电感的值如果越大&#xff0c;则电感上电流的变化量会越小。这个电流的变化量有个专用名称叫纹波电…

SVPWM所需要掌握的一些定理

1&#xff0c;正弦定理。 2.伏秒平衡&#xff08;不懂&#xff09; 伏秒平衡&#xff1a; 又称伏秒平衡&#xff0c;是指开关电源稳定工作状态下&#xff0c;加在电感两端的电压乘以导通时间等于关断时刻电感两端电压乘以关断时间&#xff0c;或指在稳态工作的开关电源中电感…