当前位置: 首页 > news >正文

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* ptrptr是要调整的内存地址(内存的起始地址也就是原来malloc和calloc已经开辟的空间的起始地址p)
  •  函数realloc的参数size_t sizesize调整之后新大小(调整新的大小需要多少),单位是字节
  •  参数size不是指新增加或减少的差距❌
  •  函数返回值是void *类型
  •  返回值为调整之后的内存起始位置
  •  返回值的类型是void* 是因为realloc函数并不知道开辟的空间类型,具体在使用的时候使用者   自己来决定。
  • 如果开辟成功,则返回一个指向开辟好空间的指针
  • 指向开辟好的空间的指针变量(分情况讨论) 

1.可能与旧空间的起始地址一致  2.可能是一块全新的空间的起始地址

 

  • 如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查
  •  realloc函数是对已经通过malloc和calloc函数开辟过的空间进行调整
  •  realloc函数的出现就是为了让动态内存管理更加灵活
  •  realloc函数在调整内存空间的是存在两种情况:
  1. 情况一:原有空间之后有足够大的空间。
  2. 情况二:原有空间之后没有足够大的空间。
  • 这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间。

关于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 常见的动态内存错误 今天收假了&#xff0c;抓紧时间写几篇博客。我又来赶进度了。今天我们来讲解动态内存管理。&#x1f197;&#x1f197; 为什么存在动态内存分配 假设我们去实现一个…...

React18入门(第二篇)——React18+Ts项目配置husky、eslint、pretttier、commitLint

前言 我的项目版本如下&#xff1a; React&#xff1a; V18.2.0Node.js: V16.14.0TypeScript&#xff1a;最新版工具&#xff1a; VsCode 本文将采用图文详解的方式&#xff0c;手把手带你快速完成在React项目中配置husky、prettier、commitLint&#xff0c;实现编码规范的统…...

【VINS】苹果手机采集单目相机+IMU数据离线运行VINS-Mono

0.准备工作 开个新坑&#xff0c;之前用Android手机做过离线采集数据的实验&#xff0c;这次用IPhone来测试&#xff01; 1.虚拟机配置Mac OS 下载一个Mac OS 的ios镜像&#xff0c;打开虚拟机按照跟Ubuntu差不多的方式安装&#xff0c;但是发现没有Mac OS的入口。 因为VMwa…...

数据结构 2.1 单链表

1.单链表 线性表&#xff1a;1.有限的序列 2.序列中的每一个元素都有唯一的前驱和后继&#xff0c;除了开头和结尾的两个节点。 顺序表&#xff1a;分配一块连续的内存去存放这些元素&#xff0c;eg、数组 链表&#xff1a;内存是不连续的&#xff0c;元素会各自被分配一块内…...

[Machine Learning]pytorch手搓一个神经网络模型

因为之前虽然写过一点点关于pytorch的东西&#xff0c;但是用的还是他太少了。 这次从头开始&#xff0c;尝试着搓出一个神经网络模型 &#xff08;因为没有什么训练数据&#xff0c;所以最后的训练部分使用可能不太好跑起来的代码作为演示&#xff0c;如果有需要自己连上数据…...

KdMapper扩展实现之Dell(pcdsrvc_x64.pkms)

1.背景 KdMapper是一个利用intel的驱动漏洞可以无痕的加载未经签名的驱动&#xff0c;本文是利用其它漏洞&#xff08;参考《【转载】利用签名驱动漏洞加载未签名驱动》&#xff09;做相应的修改以实现类似功能。需要大家对KdMapper的代码有一定了解。 2.驱动信息 驱动名称pcds…...

python和go相互调用的两种方法

前言 Python 和 Go 语言是两种不同的编程语言&#xff0c;它们分别有自己的优势和适用场景。在一些项目中&#xff0c;由于团队内已有的技术栈或者某一部分业务的需求&#xff0c;可能需要 Python 和 Go 相互调用,以此来提升效率和性能。 性能优势 Go 通常比 Python 更高效&…...

c# 分部视图笔记

Html.Partial("**", 1) public ActionResult **(int page) { ViewBag.page page; return PartialView("**"); }...

Vue3最佳实践 第七章 TypeScript 中

Vue组件中TypeScript 在Vue组件中&#xff0c;我们可以使用TypeScript进行各种类型的设置&#xff0c;包括props、Reactive和ref等。下面&#xff0c;让我们详细地探讨一下这些设置。 设置描述设置props在Vue中&#xff0c;props本身就具有类型设定的功能。但如果你希望使用Ty…...

(三)行为模式:8、状态模式(State Pattern)(C++示例)

目录 1、状态模式&#xff08;State Pattern&#xff09;含义 2、状态模式的UML图学习 3、状态模式的应用场景 4、状态模式的优缺点 &#xff08;1&#xff09;优点 &#xff08;2&#xff09;缺点 5、C实现状态模式的实例 1、状态模式&#xff08;State Pattern&#x…...

nginx的配置文件概述及简单demo(二)

默认配置文件 当安装完nginx后&#xff0c;它的目录下通常有默认的配置文件 #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 介绍&#xff1a; 近年来&#xff0c;各类恶意软件层出不穷&#xff0c;反病毒软件也更新了各种检测方案以提高检率。 其中比较有效的方案是动态沙箱检测技术&#xff0c;即通过在沙箱中运行程序并观察程序行为来判断程序是否为恶意程序。简单来说…...

QTimer类的使用方法

本文介绍QTimer类的使用方法。 1.单次触发 在某些情况下&#xff0c;定时器只运行一次&#xff0c;可使用单次触发方式。 QTimer *timer new QTimer(this); connect(timer, &QTimer::timeout, this, &MainWindow::timeout); timer->setSingleShot(true); timer-…...

(三)行为模式:9、空对象模式(Null Object Pattern)(C++示例)

目录 1、空对象模式&#xff08;Null Object Pattern&#xff09;含义 2、空对象模式的主要涉及以下几个角色 3、空对象模式的应用场景 4、空对象模式的优缺点 &#xff08;1&#xff09;优点 &#xff08;2&#xff09;缺点 5、C实现空对象模式的实例 1、空对象模式&am…...

Django实战项目-学习任务系统-用户登录

第一步&#xff1a;先创建一个Django应用程序框架代码 1&#xff0c;先创建一个Django项目 django-admin startproject mysite将创建一个目录&#xff0c;其布局如下&#xff1a;mysite/manage.pymysite/__init__.pysettings.pyurls.pyasgi.pywsgi.py 2&#xff0c;再创建一个…...

【动手学深度学习-Pytorch版】Transformer代码总结

本文是纯纯的撸代码讲解&#xff0c;没有任何Transformer的基础内容~ 是从0榨干Transformer代码系列&#xff0c;借用的是李沐老师上课时讲解的代码。 本文是根据每个模块的实现过程来进行讲解的。如果您想获取关于Transformer具体的实现细节&#xff08;不含代码&#xff09;可…...

Swift 协议扩展精进之路:解决 CoreData 托管实体子类的类型不匹配问题(下)

概述 在 Swift 开发语言中&#xff0c;各位秃头小码农们可以充分利用语法本身所带来的便利去劈荆斩棘。我们还可以恣意利用泛型、协议关联类型和协议扩展来进一步简化和优化我们复杂的代码需求。 不过&#xff0c;在涉及到多个子类派生于基类进行多态模拟的场景下&#xff0c;…...

【网络安全产品大调研系列】2. 体验漏洞扫描

前言 2023 年漏洞扫描服务市场规模预计为 3.06&#xff08;十亿美元&#xff09;。漏洞扫描服务市场行业预计将从 2024 年的 3.48&#xff08;十亿美元&#xff09;增长到 2032 年的 9.54&#xff08;十亿美元&#xff09;。预测期内漏洞扫描服务市场 CAGR&#xff08;增长率&…...

高等数学(下)题型笔记(八)空间解析几何与向量代数

目录 0 前言 1 向量的点乘 1.1 基本公式 1.2 例题 2 向量的叉乘 2.1 基础知识 2.2 例题 3 空间平面方程 3.1 基础知识 3.2 例题 4 空间直线方程 4.1 基础知识 4.2 例题 5 旋转曲面及其方程 5.1 基础知识 5.2 例题 6 空间曲面的法线与切平面 6.1 基础知识 6.2…...

实现弹窗随键盘上移居中

实现弹窗随键盘上移的核心思路 在Android中&#xff0c;可以通过监听键盘的显示和隐藏事件&#xff0c;动态调整弹窗的位置。关键点在于获取键盘高度&#xff0c;并计算剩余屏幕空间以重新定位弹窗。 // 在Activity或Fragment中设置键盘监听 val rootView findViewById<V…...

Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement

Cilium动手实验室: 精通之旅---13.Cilium LoadBalancer IPAM and L2 Service Announcement 1. LAB环境2. L2公告策略2.1 部署Death Star2.2 访问服务2.3 部署L2公告策略2.4 服务宣告 3. 可视化 ARP 流量3.1 部署新服务3.2 准备可视化3.3 再次请求 4. 自动IPAM4.1 IPAM Pool4.2 …...

c# 局部函数 定义、功能与示例

C# 局部函数&#xff1a;定义、功能与示例 1. 定义与功能 局部函数&#xff08;Local Function&#xff09;是嵌套在另一个方法内部的私有方法&#xff0c;仅在包含它的方法内可见。 • 作用&#xff1a;封装仅用于当前方法的逻辑&#xff0c;避免污染类作用域&#xff0c;提升…...

6个月Python学习计划 Day 16 - 面向对象编程(OOP)基础

第三周 Day 3 &#x1f3af; 今日目标 理解类&#xff08;class&#xff09;和对象&#xff08;object&#xff09;的关系学会定义类的属性、方法和构造函数&#xff08;init&#xff09;掌握对象的创建与使用初识封装、继承和多态的基本概念&#xff08;预告&#xff09; &a…...

C++实现分布式网络通信框架RPC(2)——rpc发布端

有了上篇文章的项目的基本知识的了解&#xff0c;现在我们就开始构建项目。 目录 一、构建工程目录 二、本地服务发布成RPC服务 2.1理解RPC发布 2.2实现 三、Mprpc框架的基础类设计 3.1框架的初始化类 MprpcApplication 代码实现 3.2读取配置文件类 MprpcConfig 代码实现…...

【无标题】湖北理元理律师事务所:债务优化中的生活保障与法律平衡之道

文/法律实务观察组 在债务重组领域&#xff0c;专业机构的核心价值不仅在于减轻债务数字&#xff0c;更在于帮助债务人在履行义务的同时维持基本生活尊严。湖北理元理律师事务所的服务实践表明&#xff0c;合法债务优化需同步实现三重平衡&#xff1a; 法律刚性&#xff08;债…...

前端开发者常用网站

Can I use网站&#xff1a;一个查询网页技术兼容性的网站 一个查询网页技术兼容性的网站Can I use&#xff1a;Can I use... Support tables for HTML5, CSS3, etc (查询浏览器对HTML5的支持情况) 权威网站&#xff1a;MDN JavaScript权威网站&#xff1a;JavaScript | MDN...