C语言中的动态内存分配

article/2025/9/14 16:38:51

大家好,今天简单讲一讲C语言中的动态内存分配。

补充:C程序中的内存块。

在C程序中,通常将内存划分为以下六个区域:

(1)内核区域。这块区域是操作系统的,用户不能使用。

(2)栈区。主要用于存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。栈内  存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。另外,当函数运行结束时,栈区的空间会被自动释放。

(3)内存映射段。该部分内存主要用于文件映射、动态库以及匿名映射。

(4)堆区。动态内存分配的区域,由程序员分配,并由程序员手动释放,若程序员不释放,则在整个程序结束后才由操作系统回收。

(5)静态区。又叫数据段,用于存放全局变量以及static修饰的变量,程序结束后由操作系统释放。

(6)代码段。用于存放函数体(类成员函数和全局函数)的二进制代码。。

截至目前,我们所使用的变量似乎都是在栈区(如局部变量、函数形参)和静态区(如全局变量、static修饰的变量),但是我们很早就知道除了栈区、静态区,在内存中还有‘堆’这样的一个区域。其实,这块内存就是用来动态内存分配的。

其实想一下就知道光靠栈区和静态区在处理问题时是很不灵活的(因为其开辟的空间是固定的),比如在声明数组时必须指定数组的大小(当然在C99中存在变长数组),但有时候我们也不能明确的指出要用多少空间,我们希望这块空间可大可小,既不过大,造成空间浪费,也不能过小,不够使用。这时就要用到我今天讲到的动态内存分配了,通过动态内存分配我们可以自由更改分配的空间,使其“可大可小”。

1.常用的动态内存分配函数(都包含在头文件<stdlib.h>中)

(1)malloc和free

malloc是最常用到的动态内存分配函数,其函数原型为:void* malloc(size_t size); 该函数向内存(堆区)申请一块连续可用空间(size个字节),并返回这块空间的地址,注意一下几点:

(1)如果内存开辟成功,该函数返回指向这块空间的指针。

(2)如果内存开辟失败,则返回空指针(NULL)。所以使用该函数时一定不要忘了进行检查,可以用assert断言,也可以像下面这样:

 (3)该函数返回void* 的指针,也就是说不知道具体类型,所以在使用时我们通常要进行强制类型转换,转换成我们需要的类型,如上例中我们需要int*的指针ptr,就将其强制转换成int*的类型。

(4)当实参size为0时,malloc的行为时C语言标准未定义的,具体取决与于编译器。

下面说一说free这个函数。

动态开辟的内存只有在程序结束时操作系统才进行回收,所以当我们不使用动态开辟的內存时必须手动将其释放掉(不然会造成内存泄漏),释放动态开辟的内存就要用到free这个函数。

其函数原型为:void  free(void*  ptr); 注意一下两点:

(1)如果实参ptr不是动态开辟的,那么该函数的行为是未定义的。

(2)如果实参ptr为空指针,那么该函数什么事都不做。

(2)calloc

calloc函数的功能与malloc类似,其函数原型为:void*  calloc(size_t  num, size_t  size); 作用是为num个大小为size的元素开辟一块空间(堆区),并且将每一个字节都初始化为0。

(3)realloc

realloc这个函数使动态内存分配更加的灵活,它可以随意调整动态开辟内存的大小。也就是说,当内存过大时可以用realloc来缩小内存,而当开辟的内存过小时可以使用realloc来扩大开辟的内存。

该函数的函数原型为:void*  realloc(void*  ptr, size_t  size); 其第一个参数ptr是之前动态开辟的内存的地址,可以是空指针(当ptr为空指针时,该函数功能与malloc完全一样),第二个参数是希望调整的内存大小(调整之后的),返回调整后的内存的地址(同样的也是void*,需要强制类型转换);如果开辟失败则返回空指针。

该函数在调整内存大小的同时,还会将内存中的数据拷贝到新的空间。其在调整内存空间时存在以下两种情况(从小调整大):

(1)在原有空间的后面有足够大的空间。这时该函数会直接在原有空间的后面扩展新的空间,原有空间的数据不会发生变化。

(2)在原有空间的后面没有足够大的空间。这时该函数会放弃原有的空间(但空间中的数据会拷贝到新的空间),在堆区另找一块合适大小的空间来使用,并且返回这块新空间的地址。

由于上述两种情况,该函数在使用时就需要注意,看下面的代码:

 在使用realloc时我们要注意上述代码中隐藏的风险。

2.常见的动态内存分配错误

动态内存分配使用方便,但是在使用时却非常容易出错,而且关于内存的错误往往都是毁灭性的。下面我列举几点最容易犯的错误,希望能帮助到大家。

(1)对空指针(NULL)解引用。看下面的代码:

 这段代码看上去似乎没什么错误,但存在潜在的风险:由于没有对指针p进行判断,所以他可能是空指针(当malloc动态内存开辟失败时),这时就会产生对空指针解引用而产生错误。

(2)对动态开辟的空间越界访问。这一点即使是存放在栈区的数组中也非常常见,不做过多解释。

(3)对非动态开辟的内存使用free释放。一定注意free只能用于释放动态开辟的空间。

(4)使用free释放动态开辟空间的一部分。

(5)对同一块动态开辟的空间多次释放。

(6)动态开辟的空间不用free释放。这样会造成内存泄漏。

(7)在函数调用后返回栈空间地址。栈空间在函数调用后就已经被销毁了,这时返回的指针就变成了野指针。

以上是使用动态内存空间常见的错误,希望大家注意,不要犯类似的错误。

3.柔性数组

在C99标准中,结构体中最后一个元素允许是未知大小的数组,这就叫做柔性数组成员。看下面的代码:

这里的int  arr[0]就是柔性数组成员。注意一下几点:

(1)结构体中的柔性数组成员前面必须至少有一个其他成员。

(2)用sizeof计算包含柔性数组成员的结构体的大小时不包括柔性数组的内存。所以上例中的sizeof(struct  Stu)为4,就是成员a的大小,没有包括柔性数组成员。

(3)包含柔性数组成员的结构体要用malloc函数进行动态内存分配,并且分配的内存应该大于该结构体的大小以适应柔性数组的预期大小。看下面的代码:

 其实上面的代码也可以这么写,和上面代码的效果是一样的:

 那么问题来了,既然不用柔性数组成员也可以达到同样的效果,那么柔性数组成员的意义何在?

仔细分析我们就能发现,使用柔性数组成员有一下两个优点:

(1)方便内存释放。使用柔性数组只进行了一次动态内存开辟,所以只用进行一次内存释放;而第二种方法进行了两次动态内存开辟,并且其中一次是对结构体内部成员进行的,如果我们的代码是在一个给别人用的函数中,你在里面做了二次内存分配,并把整个结构体返回给用户。用户调用free可以释放结构体,但是用户并不知道这个结构体内的成员也需要free,这时就会造成内存泄漏,严重时可能还会造成错误。而柔性数组成员很好的解决了这个问题。

(2)有利于提高访问速度。使用柔性数组成员其内存时一次性开辟的,是连续的一块空间,这样有利于提高访问速度,也能减少内存碎片。

好了,以上就是今天的全部内容了,水平有限,如有不足之处,还请在评论区指正;如果您觉得这篇文章对您有用,也不妨点赞收藏,谢谢!


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

相关文章

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;或指在稳态工作的开关电源中电感…

开关电源环路稳定性分析(2)-从开环到闭环

大家好&#xff0c;这里是大话硬件。 在上一节中&#xff0c;基于欧姆定律&#xff0c;基尔霍夫定律&#xff0c;伏秒平衡这些已知的知识点&#xff0c;可以推导出Buck变换器的输入输出关系。 今天这一节&#xff0c;我们还是从全局的概念来解析开关电源。 1. 运放和开关电源…

GaN图腾柱无桥 Boost PFC(单相)七-PFC占空比前馈

PFC占空比前馈概述 单纯依靠控制环路的PI调节器进行控制调节&#xff0c;控制环路的带宽会比较窄。另外&#xff0c;由于调节速度不够快&#xff0c;也将影响整个系统的动态性能。采用占空比前馈控制和反馈控制同时调节的数字控制技术&#xff0c;降低了对控制环路参数的要求。…

升压BOOST电路和降压BUCK电路最容易的理解

升压电路框图 首先在了解BOOST电路时&#xff0c;要学会如何分析最简单的升压电路框图。如图&#xff1a; 首先识别每个元件基本功能&#xff1a;电感&#xff1a;储能i电感&#xff1b;MOS管&#xff1a;开关作用&#xff1b;二极管&#xff1a;续流&#xff1b;电容&#x…

DC-DC变换器(DCDC Converter / Switched-mode Power Supply)简介

文章目录 针对B站视频&#xff0c;截取ppt做的完整笔记&#xff0c;视频地址附在下方0、DC-DC变换器概述1、DC-DC变换器的基本结构BuckBoostBuck-BoostBoost-Buck小结 2、换流与特性分析分析Buck电路分析Boost电路分析Buck-Boost电路&#xff08;前级Buck后级Boost&#xff09;…