动态内存管理(上)

目录

  • 为什么要有动态内存分配
  • 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.rhkb.cn/news/185506.html

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

相关文章

mindspore mindcv图像分类算法;模型保存与加载

参考&#xff1a; https://www.mindspore.cn/tutorials/en/r1.3/save_load_model.html https://github.com/mindspore-lab/mindcv/blob/main/docs/zh/tutorials/finetune.md 1、mindspore mindcv图像分类算法 import os from mindcv.utils.download import DownLoad import o…

一文搞定多端开发,做全栈大牛 附三大企业实战项目

一个功能三套代码 一改需求就是加不完的班&#xff1f; 不存在的&#xff0c;告别改改改 拥抱多端开发 一套代码搞定多个平台 高效开发&#xff1a;一套代码&#xff0c;多端通用 根据统计数据&#xff0c;全球移动设备用户数已经超过了50亿。随着智能手机、平板电脑等移动…

AI:68-基于深度学习的身份证号码识别

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

剑指JUC原理-15.ThreadLocal

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是爱吃芝士的土豆倪&#xff0c;24届校招生Java选手&#xff0c;很高兴认识大家&#x1f4d5;系列专栏&#xff1a;Spring源码、JUC源码&#x1f525;如果感觉博主的文章还不错的话&#xff0c;请&#x1f44d;三连支持&…

【Mac开发环境搭建】安装HomeBrew、HomeBrew安装Docker、Docker安装Mysql5.7和8

文章目录 HomeBrew安装相关命令安装包卸载包查询可用的包更新所有包更新指定包查看已经安装的包查看包的信息清理包查看brew的版本更新brew获取brew的帮助信息 Brew安装DockerDocker常用命令镜像相关查看已经拉取的所有镜像删除镜像 容器相关停止运行容器启动容器重启容器删除容…

OSPF下的MGRE实验

一、实验要求 1、R1-R3-R4构建全连的MGRE环境 2、R1-R5-R6建立hub-spoke的MGRE环境&#xff0c;其中R1为中心 3、R1-R3...R6均存在环回网段模拟用户私网&#xff0c;使用OSPF使全网可达 4、其中R2为ISP路由器&#xff0c;仅配置IP地址 二、实验拓扑图 三、实验配置 1、给各路…

自然语言处理中的文本聚类:揭示模式和见解

一、介绍 在自然语言处理&#xff08;NLP&#xff09;领域&#xff0c;文本聚类是一种基本且通用的技术&#xff0c;在信息检索、推荐系统、内容组织和情感分析等各种应用中发挥着关键作用。文本聚类是将相似文档或文本片段分组为簇或类别的过程。这项技术使我们能够发现隐藏的…

企业级低代码开发,科技赋能让企业具备“驾驭软件的能力”

科技作为第一生产力&#xff0c;其强大的影响力在各个领域中都有所体现。数字技术&#xff0c;作为科技领域中的一股重要力量&#xff0c;正在对传统的商业模式进行深度的变革&#xff0c;为各行业注入新的生命力。随着数字技术的不断发展和应用&#xff0c;企业数字化转型的趋…

10-27 maven概念

maven maven的概念模型: 项目对象模型(POM: Project object Model)&#xff0c;一组标准集合: pom.xml 依赖管理系统(Dependency Management System) 项目生命周期(Project Lifecycle) 项目对象模型&#xff1a; 把项目当成一个对象&#xff0c;描述这个项目&#xff0c;使用p…

计算机网络实验

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

Flutter利用GridView创建网格布局实现优美布局

文章目录 简介使用详解导入依赖项创建一个基本的 GridView一些参数说明使用GridView.count来构造 其他控制总结 简介 GridView 是 Flutter 中用于创建网格布局的强大小部件。它允许你在行和列中排列子小部件&#xff0c;非常适合显示大量项目&#xff0c;例如图像、文本、卡片…

【Royalty in Wind 2.0.0】个人体测计算、资料分享小程序

前言 Royalty in Wind 是我个人制作的一个工具类小程序。主要涵盖体测计算器、个人学习资料分享等功能。这个小程序在2022年第一次发布&#xff0c;不过后来因为一些原因暂时搁置。现在准备作为我个人的小程序重新投入使用XD PS&#xff1a;小程序开发部分我是在21年跟随郄培…

C# 继承,抽象,接口,泛型约束,扩展方法

文章目录 前言模拟需求场景模拟重复性高的需求初始类结构继承优化抽象类 需求1&#xff1a;打印CreateTime方法1&#xff1a;使用重载方法2&#xff1a;基类函数方法3&#xff1a;泛型约束方法3.1&#xff1a;普通泛型方法方法3.2&#xff1a;高级泛型约束&#xff0c;扩展方法…

uniapp u-tabs表单如何默认选中

首先先了解该组件&#xff1b;该组件&#xff0c;是一个tabs标签组件&#xff0c;在标签多的时候&#xff0c;可以配置为左右滑动&#xff0c;标签少的时候&#xff0c;可以禁止滑动。 该组件的一个特点是配置为滚动模式时&#xff0c;激活的tab会自动移动到组件的中间位置。 …

npm发布自己的包

npm发布自己的包 1. 首先在npm官网注册一个自己的账户(有账号的可以直接登录) 注册地址 2. 创建一个自己的项目(如果已有自己的项目, 跳过这一步) npm init -y3. 确认自己的npm下载源, 只能使用npm官方的地址 npm config get registry修改地址源 npm config set registr…

Spring Cloud学习(二)【Eureka注册中心】

文章目录 Eureka 注册中心Eureka 的作用 动手实践搭建 EurekaServer服务注册服务发现 Ribbon 负载均衡负载均衡原理IRule 接口&#xff08;负载均衡策略&#xff09;饥饿加载 Eureka 注册中心 服务调用出现的问题 不能采用硬编码服务消费者该如何获取服务提供者的地址信息&am…

苹果Mac电脑fcpx视频剪辑:Final Cut Pro中文最新 for mac

Final Cut Pro是苹果公司开发的一款专业视频剪辑软件&#xff0c;它为原生64位软件&#xff0c;基于Cocoa编写&#xff0c;支持多路多核心处理器&#xff0c;支持GPU加速&#xff0c;支持后台渲染。Final Cut Pro在Mac OS平台上运行&#xff0c;适用于进行后期制作。 Final Cu…

小白学安全-KunLun-M静态白盒扫描工具

一、KunLun-M简介 KunLun-M是一个完全开源的静态白盒扫描工具&#xff0c;支持PHP、JavaScript的语义扫描&#xff0c;基础安全、组件安全扫描&#xff0c;Chrome Ext\Solidity的基础扫描。开源地址&#xff1a;https://github.com/LoRexxar/Kunlun-M Cobra是一款源代码安全审计…

箭头函数 跟匿名函数this的指向问题

var id 10; function foo() {// 创建时 this->windowthis.id 20; // 等价于 window.id 20let c () > {console.log("id1:", this.id); // 创建时父级 创建时 this->window};let d function () {console.log("id2:", this.id); // 执行时本…

计算机毕业设计 基于SpringBoot的私人西服定制系统的设计与实现 Java实战项目 附源码+文档+视频讲解

博主介绍&#xff1a;✌从事软件开发10年之余&#xff0c;专注于Java技术领域、Python人工智能及数据挖掘、小程序项目开发和Android项目开发等。CSDN、掘金、华为云、InfoQ、阿里云等平台优质作者✌ &#x1f345;文末获取源码联系&#x1f345; &#x1f447;&#x1f3fb; 精…