动态内存管理

简介: 第一部分:为什么存在动态内存分配

本章学习目录


(❤️动态内存管理❤️)

第一部分:为什么存在动态内存分配

我们已经掌握的内存开辟方式有:

int val = 20;//在栈空间上开辟四个字节
char arr[10] = {0};//在栈空间上开辟10个字节的连续空间

但是上述的开辟空间的方式有两个特点:

  1. 空间开辟大小是固定的。
  2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配
像在创建数组的时候,可能开辟过大很多空间使用不了,可能开辟过小,空间不够使用,不够灵活,我们需要可以在内存空间中自由使用,可以变大可以变小的方法,C语言给了我们这样的权限。

我们都知道在内存分为 :
在这里插入图片描述

第二部分:动态内存函数的介绍

1. malloc

函数原型:

void* malloc (size_t size);
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针
注意:
**1.如果开辟成功,则返回一个指向开辟好空间的指针。
2.如果开辟失败,则返回一个NULL指针,因此malloc的返回值一定要做检查。**
3.返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。
4.如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器。
在这里插入图片描述

2. free

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

void free (void* ptr);

free函数用来释放动态开辟的内存。

**如果参数 ptr 指向的空间不是动态开辟的,那free函数的行为是未定义的。(free只可以释放动态开辟的空间)
如果参数 ptr 是NULL指针,则函数什么事都不做。**

malloc和free都声明在 stdlib.h 头文件中:
那么怎么使用呢?看下面代码:

在这里插入图片描述

可以看见 上面定义分别在栈区和在堆区的变量,动态内存按照书写法 前面写void*,但是会报一个警告,在一些编译器,还不可以运行,因为空类型指针,不可以解引用
在这里插入图片描述

这个函数是向内存申请了一整块内存,我们希望存放的是一块存放整形指针的数组,所以我们最好写int 类型的,根据自己的需要来更改,函数默认是void传给int* 会有警告,最好还在函数前面强制转换一下类型。
在这里插入图片描述
如果我们开辟成功,应该是下面的图:p指向新开辟的地址
在这里插入图片描述
如果没有开辟成功,那么p里面是空指针, 下面是开辟失败的情况,我们申请开辟的地址大亿点点。
在这里插入图片描述

使用这些空间的时候:

//使用这些空间的时候
    if (p == NULL)  //如果是空指针 那么就是没有开辟成功 
    {
        perror("main"); //报错提示
        return 0;
    }

使用这块空间

int i = 0;
    for ( i = 0; i < 10; i++)
    {
        *(p + i) = i;  //0-10   
    }
    for ( i = 0; i < 10; i++)
    {
        printf("%d " ,p[i]); //p[i] -> *(p+i)   也可以数组 ,
                        //因为开辟的是一块 连续的内存
    }

回收空间:申请的空间还给了操作系统。

//回收空间
    free(p);

但是p里面还是保存在以前申请的空间,会有一定非法访问内存风险,所以我们还要把p置成空指针。

p = NULL;

malloc整体使用代码:

#include <stdio.h>
#include <stdlib.h>
int main()
{
    //开辟一个10个整形的空间   10* sizeof(int)
    int arr[10]; // 局部变量  在栈区
    int* p=    (int *)malloc(10 * sizeof(int));   //在堆区
    //使用这些空间的时候
    if (p == NULL)  //如果是空指针 那么就是没有开辟成功 
    {
        perror("main"); //报错提示
        return 0;
    }
    //使用
    int i = 0;
    for ( i = 0; i < 10; i++)
    {
        *(p + i) = i;  //0-10
    }
    for ( i = 0; i < 10; i++)
    {
        printf("%d " ,p[i]); //p[i] -> *(p+i)  
    }

    //回收空间
    free(p);
    p = NULL;

    return 0;
}

以上这才是一个malloc 函数的使用,及步骤 (写代码中基本上是malloc 和free成对出现)。

3. calloc

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

void* calloc (size_t num, size_t size);
1.函数的功能是为 num 个,大小为 size 的元素开辟一块空间,并且把空间的每个字节初始化为0
2.与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

和malloc比较一下:可以看见参数不一样,calloc会初始化为0,而malloc不会
在这里插入图片描述
代码证明malloc不会初始化,calloc会初始化:

malloc 不会初始化:随机的一个地址
在这里插入图片描述
calloc会初始化:全部是0
在这里插入图片描述

4. realloc

1.realloc函数的出现让动态内存管理更加灵活。
2.有时会我们发现过去申请的空间太小了,有时候我们又会觉得申请的空间过大了,那为了合理的时候内存,我们一定会对内存的大小做灵活的调整。

那 realloc 函数就可以做到对动态开辟内存大小的调整。 函数原型如下:

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

在这里插入图片描述

原理方法:
1.返回值为调整之后的内存起始位置
2.这个函数调整原内存空间大小的基础上,还会将原来内存中的数据移动到 新 的空间
3.realloc在调整内存空间的是存在两种情况:
**(1)情况1:原有空间之后有足够大的空间
(2)情况2:原有空间之后没有足够大的空间**

怎么理解上面的话呢?相信看完下面代码及解析就懂了。

我们来个代码看看以下使用场景:
(1)首先我们申请40个int内存空间:
在这里插入图片描述
(2)我们发现我们还想要20个int内存空间,要使用realloc来开辟,这里开始解释上面的原理方法:
在这里插入图片描述

(1)情况1:原有空间之后有足够大的空间

先看看这句话怎么理解:如果原空间后面还有内存空间可以开辟,就直接返回原空间地址
在这里插入图片描述

(2)情况2:原有空间之后没有足够大的空间
在这里插入图片描述
这个时候会找新的一块,有足够空间的地方开辟,释放上面的内存,返回新内存的地址
在这里插入图片描述

上面为什么要使用新的变量,而不使用以前的变量来返回呢?
因为realloc可能会找不到空间,但是还是会返回空指针,那样的话就即丢失了以前的地址,还没有得到新的内存空间。得不偿失,所以拿一个新变量存储。

realloc单独使用也可以变成malloc
int * a= (int*)realloc(NULL, 40);
相关文章
|
物联网
IoT物联网设备OTA固件升级开发实践
通过OTA方式,我们可以对分布在全球各地的IoT设备进行设备固件升级,而不必让运维人员各地奔波。
7611 0
|
数据采集 人工智能 运维
《文档智能 & RAG让AI大模型更懂业务》解决方案体验评测
【10月更文挑战第11天】随着人工智能技术的不断进步,AI在各个领域的应用也愈发广泛。近期,我有幸接触并部署了《文档智能 & RAG让AI大模型更懂业务》这一创新性解决方案。该方案旨在通过结合文档智能处理和检索增强生成(Retrieval-Augmented Generation, RAG)技术来提升AI大模型对特定业务知识的理解能力。接下来,我将从多个角度分享我的体验与思考。
233 2
|
机器学习/深度学习 并行计算 大数据
【Python篇】深入挖掘 Pandas:机器学习数据处理的高级技巧
【Python篇】深入挖掘 Pandas:机器学习数据处理的高级技巧
321 3
|
存储 前端开发
【大前端】用html和css写一个QQ邮箱登录页面
【大前端】用html和css写一个QQ邮箱登录页面
1146 0
【大前端】用html和css写一个QQ邮箱登录页面
|
监控 网络协议 安全
2023年最新整理的中兴设备命令合集,网络工程师收藏!
2023年最新整理的中兴设备命令合集,网络工程师收藏!
1099 0
|
前端开发 Java 关系型数据库
JavaWeb用户登录注册实例(mybatis、maven、mysql、tomcat、servlet)
JavaWeb用户登录注册实例(mybatis、maven、mysql、tomcat、servlet)
JavaWeb用户登录注册实例(mybatis、maven、mysql、tomcat、servlet)
|
存储 C++
C++实现树 - 01 树的三种表示法
今天我们开始进入树的学习啦,在学习高级搜索树之前,我们先看看一些简单树的实现,先理解好概念,入门就很快了。另外,本讲将会为大家介绍双亲表示法、孩子表示法和孩子兄弟表示法的代码实现。
560 0
C++实现树 - 01 树的三种表示法
|
存储 Linux 网络安全
DVC 使用指南:外部依赖
在某些情况下,数据太大,或者其处理的组织方式使其无法在本地机器磁盘中处理,最好避免将其从当前的外部位置移动。 例如,NAS 上的数据、在 HDFS 上处理数据、通过 SSH 运行 Dask,或者用于从 S3 流式传输数据以对其进行处理的脚本。
如何查看服务id?
一 说明 服务编号也称服务id或者插件id,那么如何查看这个服务id呢?   二 如何查看 (1)登录蚂蚁开放平台【[url]https://openhome.alipay.com/platform/accountSetting.
1287 12