当前位置: 首页 > 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;可…...

HTML 列表、表格、表单

1 列表标签 作用&#xff1a;布局内容排列整齐的区域 列表分类&#xff1a;无序列表、有序列表、定义列表。 例如&#xff1a; 1.1 无序列表 标签&#xff1a;ul 嵌套 li&#xff0c;ul是无序列表&#xff0c;li是列表条目。 注意事项&#xff1a; ul 标签里面只能包裹 li…...

基于Uniapp开发HarmonyOS 5.0旅游应用技术实践

一、技术选型背景 1.跨平台优势 Uniapp采用Vue.js框架&#xff0c;支持"一次开发&#xff0c;多端部署"&#xff0c;可同步生成HarmonyOS、iOS、Android等多平台应用。 2.鸿蒙特性融合 HarmonyOS 5.0的分布式能力与原子化服务&#xff0c;为旅游应用带来&#xf…...

AI编程--插件对比分析:CodeRider、GitHub Copilot及其他

AI编程插件对比分析&#xff1a;CodeRider、GitHub Copilot及其他 随着人工智能技术的快速发展&#xff0c;AI编程插件已成为提升开发者生产力的重要工具。CodeRider和GitHub Copilot作为市场上的领先者&#xff0c;分别以其独特的特性和生态系统吸引了大量开发者。本文将从功…...

服务器--宝塔命令

一、宝塔面板安装命令 ⚠️ 必须使用 root 用户 或 sudo 权限执行&#xff01; sudo su - 1. CentOS 系统&#xff1a; yum install -y wget && wget -O install.sh http://download.bt.cn/install/install_6.0.sh && sh install.sh2. Ubuntu / Debian 系统…...

基于TurtleBot3在Gazebo地图实现机器人远程控制

1. TurtleBot3环境配置 # 下载TurtleBot3核心包 mkdir -p ~/catkin_ws/src cd ~/catkin_ws/src git clone -b noetic-devel https://github.com/ROBOTIS-GIT/turtlebot3.git git clone -b noetic https://github.com/ROBOTIS-GIT/turtlebot3_msgs.git git clone -b noetic-dev…...

嵌入式学习笔记DAY33(网络编程——TCP)

一、网络架构 C/S &#xff08;client/server 客户端/服务器&#xff09;&#xff1a;由客户端和服务器端两个部分组成。客户端通常是用户使用的应用程序&#xff0c;负责提供用户界面和交互逻辑 &#xff0c;接收用户输入&#xff0c;向服务器发送请求&#xff0c;并展示服务…...

招商蛇口 | 执笔CID,启幕低密生活新境

作为中国城市生长的力量&#xff0c;招商蛇口以“美好生活承载者”为使命&#xff0c;深耕全球111座城市&#xff0c;以央企担当匠造时代理想人居。从深圳湾的开拓基因到西安高新CID的战略落子&#xff0c;招商蛇口始终与城市发展同频共振&#xff0c;以建筑诠释对土地与生活的…...

打手机检测算法AI智能分析网关V4守护公共/工业/医疗等多场景安全应用

一、方案背景​ 在现代生产与生活场景中&#xff0c;如工厂高危作业区、医院手术室、公共场景等&#xff0c;人员违规打手机的行为潜藏着巨大风险。传统依靠人工巡查的监管方式&#xff0c;存在效率低、覆盖面不足、判断主观性强等问题&#xff0c;难以满足对人员打手机行为精…...

[ACTF2020 新生赛]Include 1(php://filter伪协议)

题目 做法 启动靶机&#xff0c;点进去 点进去 查看URL&#xff0c;有 ?fileflag.php说明存在文件包含&#xff0c;原理是php://filter 协议 当它与包含函数结合时&#xff0c;php://filter流会被当作php文件执行。 用php://filter加编码&#xff0c;能让PHP把文件内容…...

深入浅出WebGL:在浏览器中解锁3D世界的魔法钥匙

WebGL&#xff1a;在浏览器中解锁3D世界的魔法钥匙 引言&#xff1a;网页的边界正在消失 在数字化浪潮的推动下&#xff0c;网页早已不再是静态信息的展示窗口。如今&#xff0c;我们可以在浏览器中体验逼真的3D游戏、交互式数据可视化、虚拟实验室&#xff0c;甚至沉浸式的V…...