C语言之动态内存管理篇(1)
目录
为什么存在动态内存分配
动态内存函数的介绍
malloc
free
calloc
realloc
常见的动态内存错误
今天收假了,抓紧时间写几篇博客。我又来赶进度了。今天我们来讲解动态内存管理。🆗🆗
为什么存在动态内存分配
假设我们去实现一个通讯录,我们设置通讯录的大小是固定的100个元素,存放100个人的信息。如果信息太多,空间小了。如果信息太少,空间又大了。那我们应该怎样去解决?动态内存管理!
在目前为止,我们已经掌握两种向栈区申请内存的方式。
#include<stdio.h>
int main()
{int a = 10;//在栈空间申请四个字节存放一个值int arr[] = { 1,2,3,4,5,6,7,8,9,10 };//在栈空间开辟连续的空间存放一组数return 0;
}
但是上诉的开辟空间的方式有两个特点:
- 空间开辟大小是固定的
- 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。
但是对于空间的需求,不仅仅是上诉的情况。有时候我们需要的空间大小在程序运行的时候才能知道那数组的编译时开辟空间的方式就不能满足了。 这时候C语言给程序员一种权利【能够动态的申请和管理内存空间】就是【动态内存开辟】,当然除了在申请的同时 我们也要学会释放空间。
当然我们头脑中还是要有【内存分布图】

动态内存函数的介绍
接下里我们分别给大家详细介绍一下动态内存开辟的函数 我们将从:头文件 函数参数 返回值 使用等方面去介绍。大家认真学起来!!
malloc
malloc - C++ Reference (cplusplus.com)

- 头文件 #include<stdlib.h>
- 函数参数 size_t size
- 函数参数表示开辟size个字节的空间大小,单位是字节
- 如果参数size为0,malloc的行为是标准是未定义的,取决于编译器。
- 函数返回值是void *
- 如果开辟成功,则返回一个指向开辟好空间的指针
- 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
- 返回值的类型是void* 是因为malloc函数并不知道开辟的空间类型,具体在使用的时候使用者自己来决定。
- 当malloc在使用的时候,已经知道是开辟的空间是存放那种类型的数据了,可以强制类型转化
- malloc函数是向内存申请一块连续可用的空间,并返回指向这块空间的指针。
- malloc函数和free函数是配合使用的。
- malloc函数申请的空间是需要释放的。
【malloc的使用需要注意:强制类型转化&&判断&&需要free释放】
//假设现在程序员A想申请40个字节的空间去存放10个整型
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int main()
{int* p = (int*)malloc(10 * sizeof(int));//强制类型转化//开始存放if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}int i = 0;for (i = 0; i < 10; i++){p[i] = 1+i;//p相当于数组名//p+i=i+1;}//打印for (i = 0; i < 10; i++){printf("%d ", p[i]);//p+i}//释放free(p);p = NULL;return 0;//可以正常返回
}
忘记【perror库函数】戳一戳:C语言之字符函数&字符串函数篇(2)_唐唐思的博客-CSDN博客
当然如果申请的空间太大,也是不可以的!
#include<stdio.h>
#include<stdlib.h>
#include<limits.h>
int main()
{int* p = (int*)malloc(INT_MAX*4);//强制类型转化//这里的空间过大会返回NULL的if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}return 0;
}

【malloc函数申请的空间是需要释放的】要怎么释放呢?
- 主动释放:配合free函数使用。
- 被动释放:程序退出之后,malloc函数申请的空间,就会被操作系统自动回收的。
- 注意:正常情况下,谁申请的空间,谁去释放。即便不释放,也要告诉别人,让别人有机会去释放。🆗🆗很重要!
free
free - C++ Reference (cplusplus.com)

- 头文件 #include<stdilb.h>
- free函数参数是 void*ptr
- 如果参数ptr指向的空间是动态开辟的,那么free会将其释放掉。
- 如果参数ptr 指向的空间不是动态开辟,那free函数的行为是未定义的。
- 如果参数ptr 是NULL指针,则函数什么事情也不做。
- 函数没有返回值
- free函数和malloc函数是配合使用的。
- free函数专门是用来做动态内存的释放和回收的。
- 特别提醒:free完了之后空间已经被释放了,p里面任然有地址,此刻p就变成了【野指针】,所以请把p赋值为NULL(空指针)。
free(p);p = NULL;//p是接收malloc开辟的空间的起始地址的指针变量
#include <stdio.h>
int main()
{//代码1int num = 0;scanf("%d", &num);int arr[num] = { 0 };//代码2int* ptr = NULL;ptr = (int*)malloc(num * sizeof(int));if (NULL != ptr)//判断ptr指针是否为空{int i = 0;for (i = 0; i < num; i++){*(ptr + i) = 0;}}free(ptr);//释放ptr所指向的动态内存ptr = NULL;//是否有必要?return 0;
}
calloc
calloc - C++ Reference (cplusplus.com) 
- 头文件 #include<stdlib.h>
- 函数参数 size_t num size_t size (可以理解为将malloc的一个参数拆分为calloc的两个参数)
- 参数num是元素个数
- 参数size是一个元素的大小,单位是字节
- 函数返回值是 void*
- 如果开辟成功,则返回一个指向开辟好空间的指针
- 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
- 返回值的类型是void* 是因为calloc函数并不知道开辟的空间类型,具体在使用的时候使用者自己来决定。
- 当calloc在使用的时候,已经知道是开辟的空间是存放那种类型的数据了,可以强制类型转化
- calloc函数也是用来动态内存分配的
- calloc所申请的空间也需要free函数去释放
- 函数的功能为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0。
- 与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0。
//向堆区申请10个整型的空间
calloc(10, sizeof(int));
malloc(10 * sizeof(int));
除了参数的区别,calloc函数申请好空间后【会将空间初始化为0】但是malloc函数不会初始化。
#include<stdio.h>
#include<stdlib.h>
int main()
{int* p = (int*)calloc(10,sizeof(int));//强制类型转化int* p = (int*)malloc(10*sizeof(int));//强制类型转化//开始存放if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}int i = 0;//打印for (i = 0; i < 10; i++){printf("%d\n", p[i]);//p+i}//释放free(p);p = NULL;return 0;//可以正常返回
}


malloc打印出来的是随机值,而calloc打印出来是初始化为0的值。 根据需求使用,如果需要初始化为0,那我们可以使用【calloc】,如果不需要初始化为0,我们可以使用【malloc】
realloc
有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们一定会对内存的大小做灵活的调整。那 realloc 函数就可以做到对动态开辟内存大小的调整。 realloc - C++ Reference (cplusplus.com)
- 头文件 #include<stdlib.h>
- 函数realloc的参数void* ptr,ptr是要调整的内存地址(内存的起始地址也就是原来malloc和calloc已经开辟的空间的起始地址p)
- 函数realloc的参数size_t size,size调整之后新大小(调整新的大小需要多少),单位是字节
- 参数size不是指新增加或减少的差距❌
- 函数返回值是void *类型
- 返回值为调整之后的内存起始位置
- 返回值的类型是void* 是因为realloc函数并不知道开辟的空间类型,具体在使用的时候使用者 自己来决定。
- 如果开辟成功,则返回一个指向开辟好空间的指针
- 指向开辟好的空间的指针变量(分情况讨论)
1.可能与旧空间的起始地址一致 2.可能是一块全新的空间的起始地址
- 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。
- realloc函数是对已经通过malloc和calloc函数开辟过的空间进行调整
- realloc函数的出现就是为了让动态内存管理更加灵活
- realloc函数在调整内存空间的是存在两种情况:
- 情况一:原有空间之后有足够大的空间。
- 情况二:原有空间之后没有足够大的空间。
- 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间。
关于realloc函数有返回值来接收有三种不同写法。
【写法1】
p = realloc(p, 20 * sizeof(int));//用旧空间的原来的指针变量去接收(NULL问题)
#include<stdio.h>
#include<stdlib.h>
int main()
{int* p = (int*)calloc(10,sizeof(int));//强制类型转化//开始存放if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}int i = 0;//打印for (i = 0; i < 10; i++){printf("%d\n", p[i]);}//空间不够,希望调整空间为20个整型的空间p = realloc(p, 20 * sizeof(int));//不建议这样写 可能开辟空间失败返回NULL// 成功也就罢了,万一失败旧空间起始地址也找不到了//释放free(p);p = NULL;return 0;//可以正常返回
}
【写法2】
int* ptr = (int*)realloc(p, 20 * sizeof(int));//用新的指针变量去接收,但记住一定要释放
#include<stdio.h>
#include<stdlib.h>
int main()
{int* p = (int*)calloc(10,sizeof(int));//强制类型转化//开始存放if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}int i = 0;//打印for (i = 0; i < 10; i++){printf("%d\n", p[i]);}//空间不够,希望调整空间为20个整型的空间int *ptr = (int*)realloc(p, 20 * sizeof(int));//换一个指针变量去管理//如果你就是使用ptr,一定记得要释放ptr所指向的空间//释放free(p);free(ptr);p = NULL;ptr=NULL;return 0;//可以正常返回
}
【写法3】
int* ptr = (int*)realloc(p, 20 * sizeof(int));
if (ptr != NULL)
{
p = ptr;
}
#include<stdio.h>
#include<stdlib.h>
int main()
{int* p = (int*)calloc(10,sizeof(int));//强制类型转化//开始存放if (p == NULL){perror("malloc");//为空的原因return 1;//非0即不能正常返回}int i = 0;//打印for (i = 0; i < 10; i++){printf("%d\n", p[i]);}//空间不够,希望调整空间为20个整型的空间int *ptr = (int*)realloc(p, 20 * sizeof(int));//但是,程序员还是想要p来管理这块空间,可以这么写if (ptr != NULL){p = ptr;}//释放free(p);p = NULL;return 0;//可以正常返回
}
关于就是realloc函数返回的指针的两种不同的情况。
【情况1 】
要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化。
- realloc函数此时的返回值是旧的空间的起始地址

【情况2】
原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找一个合适大小的连续空间来使用。这样函数返回的是一个新的内存地址。 且realloc函数有三个特点
- realloc函数会将旧的空间的数据,拷贝到新的空间里
- realloc函数拷贝完成后,会将旧的空间释放掉
- realloc函数此刻的返回值不是原来的地址,而是新的空间的起始地址

当然除此之外,realloc还可以当成malloc来使用,只要传空指针即可。
#include<stdio.h>
#include<stdlib.h>
int main()
{int* ptr = (int*)realloc(NULL, 10 * sizeof(int));if (ptr == NULL){perror("realloc");return 1;}free(ptr);ptr = NULL;return 0;
}
常见的动态内存错误
- 对NULL指针的解引用操作
- 对动态开辟空间的越界访问
- 对非动态开辟内存使用free释放
- 使用free释放一块动态开辟内存的一部分
- 对同一块动态内存多次释放
- 动态开辟内存忘记释放(内存泄漏)
动态开辟的空间一定要正确释放!!
✔✔✔✔✔最后,感谢大家的阅读,若有错误和不足,欢迎指正!下篇博文我们讲解几道相关笔试题
代码------→【gitee:唐棣棣 (TSQXG) - Gitee.com】
联系------→【邮箱:2784139418@qq.com】
相关文章:
C语言之动态内存管理篇(1)
目录 为什么存在动态内存分配 动态内存函数的介绍 malloc free calloc realloc 常见的动态内存错误 今天收假了,抓紧时间写几篇博客。我又来赶进度了。今天我们来讲解动态内存管理。🆗🆗 为什么存在动态内存分配 假设我们去实现一个…...
React18入门(第二篇)——React18+Ts项目配置husky、eslint、pretttier、commitLint
前言 我的项目版本如下: React: V18.2.0Node.js: V16.14.0TypeScript:最新版工具: VsCode 本文将采用图文详解的方式,手把手带你快速完成在React项目中配置husky、prettier、commitLint,实现编码规范的统…...
【VINS】苹果手机采集单目相机+IMU数据离线运行VINS-Mono
0.准备工作 开个新坑,之前用Android手机做过离线采集数据的实验,这次用IPhone来测试! 1.虚拟机配置Mac OS 下载一个Mac OS 的ios镜像,打开虚拟机按照跟Ubuntu差不多的方式安装,但是发现没有Mac OS的入口。 因为VMwa…...
数据结构 2.1 单链表
1.单链表 线性表:1.有限的序列 2.序列中的每一个元素都有唯一的前驱和后继,除了开头和结尾的两个节点。 顺序表:分配一块连续的内存去存放这些元素,eg、数组 链表:内存是不连续的,元素会各自被分配一块内…...
[Machine Learning]pytorch手搓一个神经网络模型
因为之前虽然写过一点点关于pytorch的东西,但是用的还是他太少了。 这次从头开始,尝试着搓出一个神经网络模型 (因为没有什么训练数据,所以最后的训练部分使用可能不太好跑起来的代码作为演示,如果有需要自己连上数据…...
KdMapper扩展实现之Dell(pcdsrvc_x64.pkms)
1.背景 KdMapper是一个利用intel的驱动漏洞可以无痕的加载未经签名的驱动,本文是利用其它漏洞(参考《【转载】利用签名驱动漏洞加载未签名驱动》)做相应的修改以实现类似功能。需要大家对KdMapper的代码有一定了解。 2.驱动信息 驱动名称pcds…...
python和go相互调用的两种方法
前言 Python 和 Go 语言是两种不同的编程语言,它们分别有自己的优势和适用场景。在一些项目中,由于团队内已有的技术栈或者某一部分业务的需求,可能需要 Python 和 Go 相互调用,以此来提升效率和性能。 性能优势 Go 通常比 Python 更高效&…...
c# 分部视图笔记
Html.Partial("**", 1) public ActionResult **(int page) { ViewBag.page page; return PartialView("**"); }...
Vue3最佳实践 第七章 TypeScript 中
Vue组件中TypeScript 在Vue组件中,我们可以使用TypeScript进行各种类型的设置,包括props、Reactive和ref等。下面,让我们详细地探讨一下这些设置。 设置描述设置props在Vue中,props本身就具有类型设定的功能。但如果你希望使用Ty…...
(三)行为模式:8、状态模式(State Pattern)(C++示例)
目录 1、状态模式(State Pattern)含义 2、状态模式的UML图学习 3、状态模式的应用场景 4、状态模式的优缺点 (1)优点 (2)缺点 5、C实现状态模式的实例 1、状态模式(State Pattern&#x…...
nginx的配置文件概述及简单demo(二)
默认配置文件 当安装完nginx后,它的目录下通常有默认的配置文件 #user nobody; worker_processes 1;#error_log logs/error.log; #error_log logs/error.log notice; #error_log logs/error.log info;#pid logs/nginx.pid;events {worker_connection…...
Apollo Planning2.0决策规划算法代码详细解析 (2): vscode gdb单步调试环境搭建
前言: apollo planning2.0 在新版本中在降低学习和二次开发成本上进行了一些重要的优化,重要的优化有接口优化、task插件化、配置参数改造等。 GNU symbolic debugger,简称「GDB 调试器」,是 Linux 平台下最常用的一款程序调试器。GDB 编译器通常以 gdb 命令的形式在终端…...
flex 布局:元素/文字靠右
前言 略 使用flex的justify-content属性控制元素的摆放位置 靠右 <view class"more">展开更多<text class"iconfont20231007 icon-zhankai"></text></view>.more {display: flex;flex-direction: row;color: #636363;justify-co…...
java基础-第1章-走进java世界
一、计算机基础知识 常用的DOS命令 二、计算机语言介绍 三、Java语言概述 四、Java环境的搭建 JDK安装图解 环境变量的配置 配置环境变量意义 配置环境变量步骤 五、第一个Java程序 编写Java源程序 编译Java源文件 运行Java程序 六、Java语言运行机制 核心机制—Java虚拟机 核…...
jvm 堆内存 栈内存 大小设置
4种方式配置不同作用域的jvm的堆栈内存。 1、Eclise 中设置jvm内存: 改动eclipse的配置文件,对全部project都起作用 改动eclipse根文件夹下的eclipse.ini文件 -vmargs //虚拟机设置 -Xms40m //初始内存 -Xmx256m //最大内存 -Xmn16m //最小内存 -XX:PermSize=128M //非堆内…...
免杀对抗-反沙盒+反调试
反VT-沙盒检测-Go&Python 介绍: 近年来,各类恶意软件层出不穷,反病毒软件也更新了各种检测方案以提高检率。 其中比较有效的方案是动态沙箱检测技术,即通过在沙箱中运行程序并观察程序行为来判断程序是否为恶意程序。简单来说…...
QTimer类的使用方法
本文介绍QTimer类的使用方法。 1.单次触发 在某些情况下,定时器只运行一次,可使用单次触发方式。 QTimer *timer new QTimer(this); connect(timer, &QTimer::timeout, this, &MainWindow::timeout); timer->setSingleShot(true); timer-…...
(三)行为模式:9、空对象模式(Null Object Pattern)(C++示例)
目录 1、空对象模式(Null Object Pattern)含义 2、空对象模式的主要涉及以下几个角色 3、空对象模式的应用场景 4、空对象模式的优缺点 (1)优点 (2)缺点 5、C实现空对象模式的实例 1、空对象模式&am…...
Django实战项目-学习任务系统-用户登录
第一步:先创建一个Django应用程序框架代码 1,先创建一个Django项目 django-admin startproject mysite将创建一个目录,其布局如下:mysite/manage.pymysite/__init__.pysettings.pyurls.pyasgi.pywsgi.py 2,再创建一个…...
【动手学深度学习-Pytorch版】Transformer代码总结
本文是纯纯的撸代码讲解,没有任何Transformer的基础内容~ 是从0榨干Transformer代码系列,借用的是李沐老师上课时讲解的代码。 本文是根据每个模块的实现过程来进行讲解的。如果您想获取关于Transformer具体的实现细节(不含代码)可…...
RocketMQ延迟消息机制
两种延迟消息 RocketMQ中提供了两种延迟消息机制 指定固定的延迟级别 通过在Message中设定一个MessageDelayLevel参数,对应18个预设的延迟级别指定时间点的延迟级别 通过在Message中设定一个DeliverTimeMS指定一个Long类型表示的具体时间点。到了时间点后…...
Java 语言特性(面试系列1)
一、面向对象编程 1. 封装(Encapsulation) 定义:将数据(属性)和操作数据的方法绑定在一起,通过访问控制符(private、protected、public)隐藏内部实现细节。示例: public …...
STM32F4基本定时器使用和原理详解
STM32F4基本定时器使用和原理详解 前言如何确定定时器挂载在哪条时钟线上配置及使用方法参数配置PrescalerCounter ModeCounter Periodauto-reload preloadTrigger Event Selection 中断配置生成的代码及使用方法初始化代码基本定时器触发DCA或者ADC的代码讲解中断代码定时启动…...
Ascend NPU上适配Step-Audio模型
1 概述 1.1 简述 Step-Audio 是业界首个集语音理解与生成控制一体化的产品级开源实时语音对话系统,支持多语言对话(如 中文,英文,日语),语音情感(如 开心,悲伤)&#x…...
在鸿蒙HarmonyOS 5中使用DevEco Studio实现录音机应用
1. 项目配置与权限设置 1.1 配置module.json5 {"module": {"requestPermissions": [{"name": "ohos.permission.MICROPHONE","reason": "录音需要麦克风权限"},{"name": "ohos.permission.WRITE…...
如何在最短时间内提升打ctf(web)的水平?
刚刚刷完2遍 bugku 的 web 题,前来答题。 每个人对刷题理解是不同,有的人是看了writeup就等于刷了,有的人是收藏了writeup就等于刷了,有的人是跟着writeup做了一遍就等于刷了,还有的人是独立思考做了一遍就等于刷了。…...
HashMap中的put方法执行流程(流程图)
1 put操作整体流程 HashMap 的 put 操作是其最核心的功能之一。在 JDK 1.8 及以后版本中,其主要逻辑封装在 putVal 这个内部方法中。整个过程大致如下: 初始判断与哈希计算: 首先,putVal 方法会检查当前的 table(也就…...
解决:Android studio 编译后报错\app\src\main\cpp\CMakeLists.txt‘ to exist
现象: android studio报错: [CXX1409] D:\GitLab\xxxxx\app.cxx\Debug\3f3w4y1i\arm64-v8a\android_gradle_build.json : expected buildFiles file ‘D:\GitLab\xxxxx\app\src\main\cpp\CMakeLists.txt’ to exist 解决: 不要动CMakeLists.…...
探索Selenium:自动化测试的神奇钥匙
目录 一、Selenium 是什么1.1 定义与概念1.2 发展历程1.3 功能概述 二、Selenium 工作原理剖析2.1 架构组成2.2 工作流程2.3 通信机制 三、Selenium 的优势3.1 跨浏览器与平台支持3.2 丰富的语言支持3.3 强大的社区支持 四、Selenium 的应用场景4.1 Web 应用自动化测试4.2 数据…...
《Docker》架构
文章目录 架构模式单机架构应用数据分离架构应用服务器集群架构读写分离/主从分离架构冷热分离架构垂直分库架构微服务架构容器编排架构什么是容器,docker,镜像,k8s 架构模式 单机架构 单机架构其实就是应用服务器和单机服务器都部署在同一…...
