【C语言】深度解析:动态内存管理的机制与实践

请添加图片描述

🔥引言

本篇将深度解析:动态内存管理的机制。为了更加灵活分配内存中的空间,库中为了我们提供了一些的函数,去动态开辟和释放堆上的空间。

请添加图片描述
Alt

🌈个人主页:是店小二呀
🌈C语言笔记专栏:C语言笔记
🌈C++笔记专栏: C++笔记

🌈喜欢的诗句:无人扶我青云志 我自踏雪至山巅
请添加图片描述


文章目录

  • 【前文】
  • 【正文】
  • 一、动态内存开辟函数
    • 1.1 malloc
    • 1.2 calloc
    • 1.3 malloc和calloc区别
    • 1.4 realloc(动态内存扩容)
      • 1.4.1 关于realloc扩展空间的两种情况:
  • 二、free(释放动态开辟内存)
  • 三、动态内存的常见错误
  • 四、柔性数组(flexible array)
    • 4.1 柔性数组的特点
    • 4.2 柔性数组的使用
  • 五、C/C++中程序内存区域规划


【前文】

目前我们掌握申请内存的方式有两种:
int a=0;//直接开辟空间

int arr[10]={0};//连续开辟空间

上面两种开辟空间的方式存在一些问题:

  • 栈空间开辟的空间大小是固定的

  • 数组在声明时,必须指定数组的长度(一定确定大小不能被调整)

以上不能够灵活地处理内存问题,有时候是需要的空间大小在程序运行时才能知道,那么数组在编译时开辟空间的方式就不能得到满足。

对此,为了更灵活地使用空间,C语言标准库提供了程序员在堆上申请和释放空间的函数


【正文】

C语言标准库提供申请和释放动态内存空间的库函数,声明在stdlib.h头文件中。

一、动态内存开辟函数

温馨提示】:

以下三种动态内存开辟函数,都有可能会出现开辟失败的情况,对此返回值为空,通过判断指针是否为空,做出及时的处理。在OJ需要开辟空间时,一般不需要判断,一般不会开辟失败。

1.1 malloc

在这里插入图片描述

#include <stdio.h>
#include <stdlib.h>
int main()
{
    int *p=(int *)malloc(10*sizeof(int))
        if(p==NULL)
        {
            perror("malloc fail!!!")
                return 1;
        }
    free(p)
        p=NULL;
    return 0;
}

说明】:

  • 向内存申请空间不完成初始化,返回指向这块空间的大小
  • malloc是void*类型,当我们申请空间时候,需要知道申请空间交给什么类型去维护
  • 如果参数size为0,malloc可能会报错(取决于编译器)
  • 同时申请空间有时候不一定会成功。如果失败的话,将会返回一个空指针,比如申请的空间太大,就会申请失败,这一点使用的时候要去注意。

1.2 calloc

在这里插入图片描述

#include <stdio.h>
#include <stdlib.h>
int main()
{
    int *p=(int *)calloc(10,sizeof(int))
        if(p==NULL)
        {
            perror("calloc fail!!!")
                return 1;
        }
    free(p)
        p=NULL;
    return 0;
}

说明】:

  • 向内存申请空间完成初始化为0,并且返回指向这块空间的指针
  • 因为calloc是void*类型,当我们申请空间时候,需要知道申请空间交给什么类型去维护
  • 同时申请空间有时候不一定会成功。如果失败的话,将会返回一个空指针,比如申请的空间太大,就会申请失败,这一点使用的时候要去注意。

1.3 malloc和calloc区别

  • 都是向堆上申请空间
  • 参数部分不同
  • malloc申请空间没有初始化,calloc申请空间初始化为0

1.4 realloc(动态内存扩容)

在这里插入图片描述

int main()
{
    int *p=(int *)malloc(10*sizeof(int));
    if(p=NULL)
    {
        perror("malloc fail!!!");
        return 0;
    }
    int *pe=(int *)realloc(p,20*sizeof(int));
    if(pe=NULL)
    {
        perror("realloc fail!!!");
        return 0;
    }
    p=pe;
    free(p)
        p=NULL;
    pe=NULL;
    return 0;
}

说明】:

  • 申请扩展空间并返回指向扩展空间的地址

  • 一般realloc函数的使用,是在开辟好空间的基础上进行进一步的扩容

  • 如果第一个参数部分为空指针,那么realloc等价于malloc。同时需要注意是否开辟空间成功

问题】:realloc需要扩容大小,是在malloc开辟空间大小的基础上追加,还是直接申请整个空间的大小

回答】:直接申请整个空间的大小


1.4.1 关于realloc扩展空间的两种情况:

情况一】:当内存空间足够的时候,直接在申请好的空间追加

在这里插入图片描述

情况二】;当内存空间不够的时候,会在内存中寻找一块更大的空间存放,将目前的数据拷贝一份到新的空间位置中,再将原来的空间释放掉.

在这里插入图片描述


二、free(释放动态开辟内存)

在这里插入图片描述

说明】:

  • 释放动态内存空间
  • 使用方法在上面都有体现
  • free参数部分是空指针,则函数什么事都不做
  • free非动态内存就会报错重复,行为是未定义的
  • 重复释放同一块动态内存空间,会报错

注意】:如果忘记去free指针指向空间,操作系统会自动的回收使用权,但是尽量能写就写,万一出现内存泄漏危险了。


三、动态内存的常见错误

  • 对空指针的解引用操作
  • 对动态开辟的空间越界访问
  • 对非动态开辟的空间使用free释放
  • 对free释放一块动态开辟空间的一部分(空间只能一整块还)
  • 多次对一块动态内存空间使用free释放
  • 动态开辟内存忘记释放(内存泄漏)

四、柔性数组(flexible array)

在C99中,结构体最后一个成员为未知大小的数组,这个被称为柔性数组的成员,帮助用户根据要求自己给大小,更加轻松地处理可变长度的数据结构。

typedef struct st_type
{
    int i;
    int nums[0];
}type_a;
有些编译器可能会编译失败,可以化成nums[]
    typedef struct st_type
    {
        int i;
        int nums[];
    }type_a; 

4.1 柔性数组的特点

  • 结构体中至少有一个成员在柔性数组前面(如果顺序错了,也会报错)
  • sizeof返回的这种结构大小是不包含柔性数组的内存,编译器在计算结构体大小时会忽略柔性数组成员
  • 对包含柔性数组的结构体,申请空间的时候适度大于结构体的大小,以便于适应柔性数组的大小
typedef struct st_type
{
    int i;
    int a[0];//柔性数组成员
}type_a;
int main()
{
    printf("%d\n", sizeof(type_a));//输出的是4
    return 0;
}

4.2 柔性数组的使用

代码一】:

typedef struct st_type
{
    int i;
    int *p_a;
}type_a;
int main() 
{
    int i = 0;
    type_a *p = (type_a*)malloc(sizeof(type_a)+100*sizeof(int));
    p->i = 100;

    for(i=0; i<100; i++)
    {
        p->a[i] = i;
    }
    free(p);
    return 0;
}

代码二】:

typedef struct st_type
{
    int i;
    int *p_a;
}type_a;
int main()
{
    type_a *p = (type_a *)malloc(sizeof(type_a));
    p->i = 100;
    p->p_a = (int *)malloc(p->i*sizeof(int));

    //业务处理
    for(i=0; i<100; i++)
    {
        p->p_a[i] = i;
    }

    //释放空间
    free(p->p_a);
    p->p_a = NULL;
    free(p);
    p= NULL;
    return 0;
}

说明】:

  • 就是在一块空间内再开辟一块空间使用。
  • 第一种和第二种都能实现相同的效果,但是第一种有两个好处

【第一个好处】:

  • 如果里面做了二次内存分配,并把整个结构体返回给用户。当用户需要释放空间时候,并不知道这个结构体内成员也需要free。
  • 如果结构体的内存以及其成员要的内存一次性分配好,返回一个结构体指针,用户只需要一次free就可以把所有的内存也给释放掉了

【第二个好处】:连续的内存有益于提高访问速度,也有益于减少内存碎片


五、C/C++中程序内存区域规划

请添加图片描述

内存分配的几个区域:

栈区(stack):

  • 在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。
  • 栈内存分配运算内置于处理器的指令集中,效率很高,但是分的内存容量有限
  • 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等

堆区(heap)
一般由程序员分配释放,若程序员不释放,程序结束时可能由OS(操作系统)回收。

数据段(静态区):(static)存放全局变量、静态数据。程序结束后由系统释放。

代码段:存放函数体(类成员函数和全局函数)的二进制代码


请添加图片描述

以上就是本篇文章的所有内容,在此感谢大家的观看!这里是店小二C语言笔记,希望对你在学习C语言中有所帮助!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【Redis】Redis入门概述

&#x1f600;大家好&#xff0c;我是白晨&#xff0c;一个不是很能熬夜&#x1f62b;&#xff0c;但是也想日更的人✈。如果喜欢这篇文章&#xff0c;点个赞&#x1f44d;&#xff0c;关注一下&#x1f440;白晨吧&#xff01;你的支持就是我最大的动力&#xff01;&#x1f4…

选择数据摆渡系统 哪个厂家比较好?

数据摆渡系统是一种用于在不同网络或安全域之间安全传输数据的技术解决方案。它通常用于解决网络隔离环境下的数据交换问题&#xff0c;确保数据在传输过程中的安全性、完整性和合规性。数据摆渡系统可以采用物理手段&#xff0c;如通过专用隔离硬件和私有安全协议&#xff0c;…

数据结构的二叉树(c语言版)

一.二叉树的概念 1.二叉树的基本概念 二叉树是一种常见的树状数据结构&#xff0c;它由若干个节点组成&#xff0c;这些节点通过边连接起来。每个节点最多可以有两个子节点&#xff0c;分别称为左子节点和右子节点。 二叉树的特点是每个节点最多有两个子节点&#xff0c;而且…

上海市青少年算法2023年12月月赛(丙组)试题解析

上海市青少年算法2023年12月月赛(丙组)试题解析 T1数砖数 题目描述 给定一种 22 规格的瓷砖,该瓷砖的式样为 ## .# 用这种瓷砖,从平面的左上角出发,将整个平面铺满,形如: 给定两个整数 n 与 m,请计算从左上角开始的 n 行 m 列的区域中,有多少格子是 #。 输入格式 第一…

易百纳与成都鼎桥达成战略合作,共创海鸥派生态新篇章

前言 易百纳技术社区&#xff08;以下简称“易百纳”&#xff09;与成都鼎桥通信技术有限公司&#xff08;以下简称“成都鼎桥”&#xff09;达成共建海鸥派的周边生态战略合作。基于海鸥派推出的鼎桥TD OS嵌入式发行版1.0&#xff0c;为海鸥派生态注入新的活力。 易百纳 Eba…

机器人增量学习研究综述

源自&#xff1a;控制与决策 作者&#xff1a;马旭淼 徐德 “人工智能技术与咨询” 发布 摘 要 机器人的应用场景正在不断更新换代,数据量也在日益增长.传统的机器学习方法难以适应动态的环境,而增量学习技术能够模拟人类的学习过程,使机器人能利用旧知识来加快新任务的…

数据分析处理的步骤是什么?制造业企业如何挑选数据分析处理软件?看这篇就够了

随着工业4.0的深入实施以及国家对制造业高质量发展战略的日益强调&#xff0c;工业数据已经崭露头角&#xff0c;成为生产经营活动中至关重要的核心要素。不仅如此&#xff0c;工业数据还作为优质的生产要素&#xff0c;为新兴生产力的形成提供了强有力的支撑&#xff0c;从而推…

docker自建GitLab仓库

摘要 GitLab 是一个功能强大的开源代码托管平台&#xff0c;它不仅提供了代码存储和版本控制的核心功能&#xff0c;还集成了项目管理、CI/CD 流水线、代码审查等企业级特性。本文将指导你如何在自己的服务器上搭建 GitLab 社区版&#xff0c;创建一个完全属于自己的开源仓库&…

Linux基础命令(续)

17&#xff0c;wc命令 作用&#xff1a;统计行数、单词数、字符个数 格式&#xff1a; wc 选项 文件 wc passwd 26 36 1159 passwd26&#xff1a;行数 36&#xff1a;单词数 1159&#xff1a;字符数 passwd&#xff1a;文件名wc autofs.conf 426 2604 15137 autofs.conf426…

某攻防演练心得之随笔记

最近太忙了&#xff0c;忙于各种奇奇怪怪的事情&#xff0c;有攻防&#xff0c;有应急&#xff0c;有渗透&#xff0c;还成为了一段时间内的“word高级工程师”......有师傅说我现在更新的越来越慢了&#xff0c;是呀&#xff0c;其实我也不知道怎么了&#xff0c;每天各种新闻…

用balenaEtcher烧录ubuntu的iso文件都失败,所以选用了另一种烧录的软件Rufus,然后烧录成功了+安装ubuntu的坑

https://releases.ubuntu.com/bionic/进入网页下载ubuntu 选择烧录软件将下载的Ubuntu烧录到U盘中 之前用这个U盘烧录过一次&#xff0c;成功了&#xff0c;后来应该是U盘受损或者是什么其他原因使得用这个U盘总是烧录失败 换思路&#xff1a;由于一直使用balenaEtcher烧录ubu…

《四月女友》开启预售 “不想错过”鼓励情侣找回消失的爱

电影《四月女友》由中国电影集团公司进口&#xff0c;中国电影股份有限公司发行、译制&#xff0c;改编自川村元气同名小说&#xff0c;山田智和导演&#xff0c;佐藤健、长泽雅美、森七菜主演。《四月女友》今日发布“不想错过”版预告&#xff0c;预告中&#xff0c;佐藤健饰…

【论文阅读笔记】HermesSim(Code is not Natural Language) (Security 24)

个人博客地址 HermesSim [Security 24] 论文&#xff1a;《Code is not Natural Language: Unlock the Power of Semantics-Oriented Graph Representation for Binary Code Similarity Detection》 仓库&#xff1a;https://github.com/NSSL-SJTU/HermesSim 提出的问题 二…

Pyhton专题学习资料包,Python从入门到精通全套学习资料[30G]

资源概览 百本Python学习书籍大礼包百本前端学习书籍大礼包微专业-数据挖掘分析之Python篇小甲鱼零基础入门学习Python(全96集) 资源获取 &#x1f9d1;‍&#x1f4bb;【Pyhton专题资料】【30G】 百本Python书籍## 百本前端书籍 微专业-数据挖掘分析之Python篇 预备课【先…

【问题解决】记一个“奇怪”的java.lang.NoSuchMethodError错误

问题现象 近期本人负责的一个SpringBoot模块出现了java.lang.NoSuchMethodError报错&#xff0c;问题情况如下&#xff1a; A类提供了setJumpType(String type)&#xff0c;B类调用A类的setJumpType(String type)报错java.lang.NoSuchMethodError: com.xxx.A.setJumpType(Lja…

北京软考职称、入户相关疑问解答

很多考生考软考证书是为了职称和入户&#xff0c;那么北京软考证书可以用于入户吗&#xff1f;软考职称认定有什么条件&#xff1f; 北京软考职称如何认定&#xff1f; 从以上图片描述我们可以知道&#xff0c;北京软考职称认定是有学历、资历要求的。北京近两年的“职称评审通…

HTTP 连接详解

概述 世界上几乎所有的 HTTP 通信都是由 TCP/IP 承载的&#xff0c;客户端可以打开一条TCP/IP连接&#xff0c;连接到任何地方的服务器。一旦连接建立&#xff0c;客户端和服务器之间交换的报文就永远不会丢失、受损或失序 TCP&#xff08;Transmission Control Protocol&…

防火墙技术基础篇:网络地址转换(NAT):防火墙技术的核心机制

防火墙技术基础篇&#xff1a;网络地址转换&#xff08;NAT&#xff09;&#xff1a;防火墙技术的核心机制 网络地址转换&#xff08;NAT&#xff09;是现代网络架构中不可或缺的一个组成部分&#xff0c;尤其在防火墙技术的实现中扮演着重要角色。本文旨在全面解读NAT的工作机…

pci设备枚举流程

概念 PCI设备&#xff1a;遵循PCI规范&#xff0c;工作在PCI局部总线环境下的设备。PCI局部总线规范指出&#xff0c;每个PCI设备可以包含最多8个PCI功能&#xff0c;每个PCI功能是一个逻辑设备 PCI桥设备&#xff1a;由于电子负载限制&#xff0c;每条PCI总线上可以挂载的设…

机器人学导论实验3-机器人定位中的直线拟合与提取

目录 1 实验目的 2 任务一&#xff1a;直线拟合 2.1 内容分析 2.2 过程分析 2.3 结果分析 3 任务二&#xff1a;直线提取 3.1 内容分析 3.2 过程分析 3.3 结果分析 4 遇到的问题和心得 机器人导论实验-机器人定位中的直线拟合与提取 1 实验目的 2 任务一&#xff1a; 直线…