动态内存管理(上)

目录

  • 为什么要有动态内存分配
  • malloc和free
    • malloc
    • free
  • calloc和realloc
  • calloc
    • realloc

感谢各位大佬对我的支持,如果我的文章对你有用,欢迎点击以下链接
🐒🐒🐒 个人主页
🥸🥸🥸 C语言
🐿️🐿️🐿️ C语言例题
🐣🐓🏀 python

为什么要有动态内存分配

在学动态内存分布时我们先了解一下内存的一点知识
在这里插入图片描述
内存是分栈区,堆区和静态区

我们已经掌握的内存开辟⽅式有
int val = 20;//在栈空间上开辟四个字节
char arr[10] = {0};//在栈空间上开辟10个字节的连续空间

但是上述的开辟空间的方式有两个特点:
空间开辟大小是固定的
数组在申明的时候,必须指定数组的长度,数组空间一旦确定了大小不能调整

但是对于空间的需求,不仅仅是上述的情况。
有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了。

而如果我们一次性就开辟很大的空间,比如我只明明只需要开辟4个字节的空间,而最后却开辟了100个字节的空间,那么多出来的空间就浪费掉了

因此C语言引人了动态内存开辟,让程序员自己可以申请和释放空间,就比较灵活了。(注意必须最后要释放,如果只申请不释放的话内存会越来越大)

malloc和free

malloc

C语言提供了一个动态内存开辟的函数malloc:

void* malloc (size_t size);

这个函数向内存申请一块连续可用的空间,并返回指向这块空间起始地址的指针
如果开辟成功,则返回一个指向开辟好空间的指针
如果开辟失败,则返回一个 NULL 指针,因此malloc的返回值一定要做检查(用perror(“malloc”)来检查)
返回值的类型是 void ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定*。
如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器
什么情况会使内存开辟失败呢?
来看看下面的一个代码

#include<stdio.h>
#include<limits.h>
#include<stdlib.h>
int main()
{int* p = (int*)malloc(INT_MAX*10);if (p == NULL){perror("malloc");return 1;//非0表示异常返回}return 0;//0表示正常返回
}

INT_MAX是整形的最大值,他的值如图
在这里插入图片描述
在我们还没有运行的时候系统就已经报错了
在这里插入图片描述
我们再看看perror输出的结果

在这里插入图片描述
显然我们开辟的空间过大,导致没有了足够的空间

如果我们申请0个字节会怎么样
在这里插入图片描述
并没有报错
那我们再申请-1个字节又会怎么样
在这里插入图片描述
我们可以看到输出的结果是没有足够的空间

我们再看一个代码

#include<stdio.h>
#include<limits.h>
#include<stdlib.h>
int main()
{int* p = (int*)malloc(10*sizeof(int));if (p == NULL){perror("malloc");return 1;}for (int i = 0; i < 10; i++){*(p + i) = i;}for (int i = 0; i < 10; i++){printf("%d ", p[i]);}return 0;}

在这里插入图片描述
这里的p指向malloc开辟的起始地址,由于内存中存放的都是int类型的元素,那么p+i就表示p跳过i个元素,对其解引用就是内存中的具体值,这里就非常像数组
在这里插入图片描述

free

C语言提供了另外一个函数free,专门是用来做动态内存的释放和回收的,函数原型如下

void free (void* ptr);

free函数用来释放动态开辟的内存
如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的
如果参数 ptr 是NULL指针,则函数什么事都不做
malloc和free都声明在 stdlib.h 头文件中
在之前有提到,每次都要释放内存,而在malloc的代码中却没有释放,这里需要解释一下

#include<stdio.h>
#include<limits.h>
#include<stdlib.h>
int main()
{int* p = (int*)malloc(10*sizeof(int));if (p == NULL){perror("malloc");return 1;}for (int i = 0; i < 10; i++){*(p + i) = i;}for (int i = 0; i < 10; i++){printf("%d ", p[i]);}return 0;}

由于这个程序是运行完了的,所以系统会自动回收内存,因此并不需要我们主动去回收(当然了主动回收也是一个好习惯),而有些程序是一直运行的,或者会运行很久,这样导致开辟的空间一直没有释放,而在运行的过程中不断的开辟新的内存空间,导致内存越来越大,所以需要自己去主动释放内存

最后结论就是如果程序运行完,就会自己回收内存空间,这是被动回收
而用free回收的话,就是可以在程序还没有运行完就可以回收,这是主动回收

举个例子:

#include <stdio.h>
#include <stdlib.h>
int main()
{int num = 0;scanf("%d", &num);int arr[num] = { 0 };int* 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 = NULL;return 0;
}

在这里插入图片描述
这里free时我们只是回收了arr的内存空间,但是ptr却仍然保存着arr的地址,因此在最后我们需要让ptr==NULL变成空指针,否则就会变成野指针

calloc和realloc

calloc

C语言还提供了一个函数叫 calloc , calloc 函数也用来动态内存分配。原型如下

void* calloc (size_t num, size_t size);

函数的功能是为 num 个大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0

与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为0
举个例子:

#include <stdio.h>
#include <stdlib.h>
int main()
{int* p = (int*)calloc(10, sizeof(int));//10表示申请多少个元素,sizeof(int)表示申请一个元素有多大if (NULL != p){int i = 0;for (i = 0; i < 10; i++){printf("%d ", *(p + i));}}free(p);p = NULL;return 0;
}

在这里插入图片描述
由于calloc初始化,使空间中的元素全为0,所以输出的结果都是0

#include <stdio.h>
#include <stdlib.h>
int main()
{int* p = (int*)malloc(10*sizeof(int));if (NULL != p){int i = 0;for (i = 0; i < 10; i++){printf("%d ", *(p + i));}}free(p);p = NULL;return 0;
}

在这里插入图片描述
malloc没有初始话,导致输出的结果就是这样

所以如果我们对申请的内存空间的内容要求初始化,那么可以很方便的使用calloc函数来完成任务

realloc

realloc函数的出现让动态内存管理更加灵活。

有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们⼀定会对内存的大小做灵活的调整

那 realloc 函数就可以做到对动态开辟内存大小的调整

函数原型如下

void* realloc (void* ptr, size_t size);

ptr 是要调整的内存地址
size 调整之后新大小
返回值为调整之后的内存起始位置
这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到新的空间
realloc在调整内存空间的是存在两种情况
◦ 情况1:原有空间之后有足够大的空间
◦ 情况2:原有空间之后没有足够大的空间

情况1
当是情况1的时候,要扩展内存就直接原有内存之后直接追加空间,原来空间的数据不发生变化
在这里插入图片描述

情况2
当是情况2的时候,原有空间之后没有足够多的空间时,扩展的方法是:在堆空间上另找⼀个合适大小
的连续空间来使用。这样函数返回的是一个新的内存地址

在这里插入图片描述
我们具体举个例子

#include <stdio.h>
#include <stdlib.h>
int main()
{int* ptr = (int*)malloc(100 * sizeof(int));//扩展容量 int* p = NULL;p = realloc(ptr, 10000 * sizeof(int));ptr = p;free(ptr);return 0;
}

在这里插入图片描述
在这里插入图片描述
显然这里的realloc是重新找了一块新的内存空间,并返回的

因此由于上述的两种情况,realloc函数的使用就要注意⼀些。

//代码1 
#include <stdio.h>
#include <stdlib.h>
int main()
{int* ptr = (int*)malloc(100);if (ptr != NULL){//业务处理} else{return 1;} //扩展容量ptr = (int*)realloc(ptr, 1000);//这样可以吗?(如果申请失败会如何?)	free(ptr);return 0;
}
//代码2
#include <stdio.h>
#include <stdlib.h>
int main()
{int* ptr = (int*)malloc(100*sizeof(int));if (ptr != NULL){//业务处理} else{return 1;} //扩展容量 int* p = NULL;p = realloc(ptr, 1000*sizeof(int));if (p != NULL){ptr = p;} free(ptr);return 0;
}

代码一和代码二的区别就在于一个是直接用ptr接收realloc的返回值
一个是判断realloc的返回值是否为空
为什么需要判断返回值是否为空呢?

如果返回值为空,那么让ptr等于空指针的话,之前内存所保存的所有元素都被消失(也可以说是之前申请的内存都没有了,当然就没有保存的元素了)

而如果判断返回值不为空之后,再将返回值传给ptr,那么内存保存的所以元素都不会消失

此外realloc还可以当成malloc使用

#include<stdio.h>
int main()
{realloc(NULL,sizeof(int))//==malloc(1*sizeof(int))return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/135960.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

有关python库

官方库 #1、导入某模块 import os #2、导入OS模块中的system方法 from os import system #3、导入某模块中的孙子模块中的xx方法&#xff0c;并重命名 from module.xx.xx import xx as rename #4、导入OS中的所有模块 #不用进行OS.method(),直接method&#xff08;&#xff0…

RRC configured BWP

TS 38.822有UE BWP 相关能力 IE的详细介绍,如下图。 举例说明,对于UE上报bwp-SameNumerology=upto2时,根据上图中的描述,UE支持能力情况如下:每个carrier最多支持2 个UE specific RRC configured DL/UL BWPs;可以通过DCI和BWP-InactivityTimer主动切换BWP;每个carrier的…

Linux安装Python3.10与部署flask项目实战详细记录

java开发新手入门Python,创建flask后端服务对外提供访问.记录一下在阿里云服务器部署flask项目的操作过程,简单介绍一下使用的阿里云服务器系统配置:ubantu16.04,其他内核版本操作部分命令会有所区别,下面开始详细操作过程! 1.pycharm创建flask项目并打包 2.Python3.…

【React】04.MVC模式和MVVM模式

React是Web前端框架 1、目前市面上比较主流的前端框架 ReactAngular&#xff08;NG框架&#xff09;Vue 主流的思想&#xff1a; 不在直接去操作DOM&#xff0c;而是改为“数据驱动思想” 操作DOM思想&#xff1a; 操作DOM比较消耗性能[主要原因就是&#xff0c;可能会导…

基于开源项目OCR做一个探究(chineseocr_lite)

背景&#xff1a;基于图片识别的技术有很多&#xff0c;应用与各行各业&#xff0c;我们公司围绕电子身份证识别自动录入需求开展&#xff0c;以下是我的研究心得 技术栈&#xff1a;python3.6&#xff0c;chineseocr_lite的onnx推理 环境部署&#xff1a;直接上截图&#xff…

Linux shell编程学习笔记22: () $() (()) 的用法小结

最近学习Linux Shell编程&#xff0c;对 () (()) [] [[]]等符号的用法还是有点分不太清楚&#xff0c;于是决定再梳理一下。今天先整理 () $() (()) 的用法。 1 单小括号() 1.1 子shell&#xff08;命令组&#xff09; 括号中的命令将会新开一个子shell顺序执行&#xff0c;所…

基于nginx在视频播放器与服务器之间反向代理流程

1 服务器部署 由于我手里只有内网服务器&#xff0c;可以使用&#xff0c;因此在部署nginx代理服务器&#xff0c;使之在播放器和服务器之间实现反向代理并且缓存内容之前&#xff0c;需要做内网穿透&#xff0c;获得可与外界进行通信的地址。 如果想进行内网穿透&#xff0c;…

使用 AIGC ,ChatGPT 快速合并Excel工作薄

职场数据处理&#xff0c;数据分析汇报与统计的过程中&#xff0c;经常会遇到这样的一个问题那就是需要统计的数据源在多个文件中&#xff0c;多个工作薄中&#xff0c;如果要进行数据处理&#xff0c;汇总的时候会很不方便 例如&#xff1a; 如果要汇总6个月的数据可能就得需…

初步利用Ansible实现批量服务器自动化管理

1.Ansible介绍 Ansible是一款开源的自动化运维工具, 在2012年由Michael DeHaan创建, 现在由Red Hat维护。Ansible是基于Python开发的,采用YAML语言编写自动化脚本playbook, 可以在Linux、Unix等系统上运行, 通过SSH协议管理节点, 无需在被管理节点安装agent。Ansible以其简单、…

6-爬虫-scrapy解析数据(使用css选择器解析数据、xpath 解析数据)、 配置文件

1 scrapy解析数据 1.1 使用css选择器解析数据 1.2 xpath 解析数据 2 配置文件 3 整站爬取博客–》爬取详情–》数据传递 scrapy 爬虫框架补充 # 1 打码平台---》破解验证码-数字字母&#xff1a;ddddocr-计算题&#xff0c;滑块&#xff0c;成语。。。-云打码&#xff0c;超…

AI:69-基于深度学习的音乐推荐

🚀 本文选自专栏:AI领域专栏 从基础到实践,深入了解算法、案例和最新趋势。无论你是初学者还是经验丰富的数据科学家,通过案例和项目实践,掌握核心概念和实用技能。每篇案例都包含代码实例,详细讲解供大家学习。 📌📌📌在这个漫长的过程,中途遇到了不少问题,但是…

索引和事务

文章目录 一、索引1.1 概念1.2 作用1.3 使用场景1.4 使用 二、事务2.1 为什么要使用事务2.2 事务的概念2.3 事务的使用 三、内容重点总结 一、索引 1.1 概念 索引是一种特殊的文件&#xff0c;包含对数据表所有记录的引用指针。在MySQL中&#xff0c;索引是基于一个或多个列的…

【紫光同创国产FPGA教程】——PDS安装教程

本原创教程由深圳市小眼睛科技有限公司创作&#xff0c;版权归本公司所有&#xff0c;如需转载&#xff0c;需授权并注明出处 一&#xff1a;软件简介 PangoDesign Suite是紫光同创基于多年FPGA开发软件技术攻关与工程实践经验而研发的一款拥有国产自主知识产权的大规模FPGA开…

Redis系列-Redis过期策略以及内存淘汰机制【6】

目录 Redis系列-Redis过期策略以及内存淘汰机制【6】redis过期策略内存淘汰机制算法LRU算法LFU 其他场景对过期key的处理FAQ为什么不用定时删除策略? Ref 个人主页: 【⭐️个人主页】 需要您的【&#x1f496; 点赞关注】支持 &#x1f4af; Redis系列-Redis过期策略以及内存淘…

18 CDN详解

1、理解CDN 1.CDN 和电商系统的分布式仓储系统一样&#xff0c;就近发货给客户(客户端)&#xff0c;所以&#xff0c;必然是提前在仓库中存储了某些商品. 2.CDN最擅长的是缓存静态数据&#xff0c;比如电商系统的热点静态页面&#xff0c;秒杀场景的页面等.问题&#xff1a;向…

快手快速涨粉的方法,自动涨粉软件的开发分享与实操分享

先来看视频实操成果&#xff0c;↑↑需要的同学可看我名字↖↖↖↖↖&#xff0c;或评论888无偿分享 一、引言 随着互联网的飞速发展&#xff0c;快手已经成为了许多人分享生活、展示才艺的平台。在快手上&#xff0c;如果你想要快速涨粉&#xff0c;就需要掌握一些技巧和方法…

计算机网络实验

计算机网络实验 使用软件PT7.0按照上面的拓扑结构建立网络&#xff0c;进行合理配置&#xff0c;使得所有计算机之间能够互相通信。并且修改各交换机的系统名称为&#xff1a;学号_编号&#xff0c;如你的学号为123&#xff0c;交换机Switch0的编号为0&#xff0c;则系统名称为…

前端Vue 页面滑动监听 拿到滑动的坐标值

前言 前端Vue 页面滑动监听 拿到滑动的坐标值 实现 Vue2写法 mounted() {// 监听页面滚动事件window.addEventListener("scroll", this.scrolling);}, methods: { scrolling() {// 滚动条距文档顶部的距离let scrollTop window.pageYOffset ||document.documentE…

如何使用Node.js快速创建HTTP服务器并实现公网访问本地Server

文章目录 前言1.安装Node.js环境2.创建node.js服务3. 访问node.js 服务4.内网穿透4.1 安装配置cpolar内网穿透4.2 创建隧道映射本地端口 5.固定公网地址 前言 Node.js 是能够在服务器端运行 JavaScript 的开放源代码、跨平台运行环境。Node.js 由 OpenJS Foundation&#xff0…

Redis系列-Redis数据类型【3】

目录 Redis系列-Redis数据类型【3】字符串类型&#xff08;String&#xff09;SDS (simple dynamic string) 哈希类型&#xff08;Hash&#xff09;列表类型&#xff08;List&#xff09;集合类型&#xff08;Set&#xff09;有序集合类型&#xff08;ZSet&#xff09;字符串类…