C语言动态内存的管理
前言
本篇博客就来探讨一下动态内存,说到内存,我们以前开辟空间大小都是固定的,不能调整这个空间大小,于是就有动态内存,可以让我们自己选择开辟多少空间,更加方便,让我们一起来看看动态内存的有关知识吧
个人主页:小张同学zkf
若有问题 评论区见
感兴趣就关注一下吧
目录
1.什么是动态内存
2. malloc和free
2.1 malloc
2.2 free
3. calloc和realloc
3.1 calloc
3.2 realloc
4. 常见的动态内存的错误
4.1 对NULL指针的解引用操作
4.2 对动态开辟空间的越界访问
4.3 对非动态开辟内存使用free释放
4.4 使用free释放一块动态开辟内存的一部分
4.5 对同一块动态内存多次释放
4.6 动态开辟内存忘记释放(内存泄漏)
5. 动态内存经典笔试题分析
5.1 题目1:
5.2 题目2:
5.3 题目3:
5.4 题目4:
6. 柔性数组
6.1 柔性数组的特点:
6.2 柔性数组的使用
6.3 柔性数组的优势
1.什么是动态内存
首先我们要搞清楚什么是动态内存的分配?

平常我们定义的数组,都是在栈区分配的空间,都是分配的空间都是固定的大小
这种分配固定大小的内存分配方法称之为静态内存分配
与静态内存相对的,就是可以控制内存的分配的动态内存分配
注意:这里动态内存分配的空间是在堆区申请的,不是在栈区申请的
我们再来看看内存各个空间都是什么

2. malloc和free
我们来了解下动态内存的函数,对了以下所有函数的头文件都是<stdlib.h>
2.1 malloc
C语言提供了一个动态内存开辟的函数:
void * malloc ( size_t size);
这个函数向内存申请一块连续可用的空间,并返回指向这块空间的指针。
注意:
• 如果开辟成功,则返回一个指向开辟好空间的指针。• 如果开辟失败,则返回一个 NULL 指针,因此malloc的返回值一定要做检查。• 返回值的类型是 void* ,所以malloc函数并不知道开辟空间的类型,具体在使用的时候使用者自己来决定。• 如果参数 size 为0,malloc的行为是标准是未定义的,取决于编译器
2.2 free
void free ( void * ptr);
#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所指向的动态内存ptr = NULL;//是否有必要?return 0;
}
看这个例子就是典型的动态内存的开辟和回收,malloc开辟空间,然后判断一下是不是开辟空间失败,若失败返回空指针,当动态内存你使用完毕之后,用free释放,释放后的指针是野指针,记得置空。
3. calloc和realloc
3.1 calloc
void * calloc ( size_t num, size_t size);

#include <stdio.h>
#include <stdlib.h>
int main()
{int *p = (int*)calloc(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;
}
输出结果:
0 0 0 0 0 0 0 0 0 0
所以如果我们对申请的内存空间的内容要求初始化,那么可以很方便的使用calloc函数来完成任务。
3.2 realloc
void * realloc ( void * ptr, size_t size);

#include <stdio.h>
#include <stdlib.h>
int main()
{int *ptr = (int*)malloc(100);if(ptr != NULL){//业务处理}else{return 1; }//扩展容量//代码1 - 直接将realloc的返回值放到ptr中ptr = (int*)realloc(ptr, 1000);//这样可以吗?(如果申请失败会如何?)//代码2 - 先将realloc函数的返回值放在p中,不为NULL,在放ptr中int*p = NULL;p = realloc(ptr, 1000);if(p != NULL)
{ptr = p;}//业务处理free(ptr);return 0;
}
realloc在vs上,是情况2的情况,自动释放旧的动态空间,在新的动态空间里开辟更大的空间,自动把就空间的数据拷贝一份到新空间,返回新空间的初始指针,所以不用再用free释放旧空间,只需释放realloc开批的新空间,记住realloc开辟的新空间也有可能开辟失败,若开辟失败,返回空指针。
4. 常见的动态内存的错误
4.1 对NULL指针的解引用操作
void test (){int *p = ( int *) malloc (INT_MAX/ 4 );*p = 20 ; // 如果 p 的值是 NULL ,就会有问题free (p);}
看这个代码,这个动态内存开辟的空间没有判断p是不是空指针,有可能内存开辟失败返回空指针,若对空指针解引用,就会非法访问出错。
4.2 对动态开辟空间的越界访问
void test (){int i = 0 ;int *p = ( int *) malloc ( 10 * sizeof ( int ));if ( NULL == p){exit (EXIT_FAILURE);}for (i= 0 ; i<= 10 ; i++){*(p+i) = i; // 当 i 是 10 的时候越界访问}free (p);}
仔细看这个i,当它等于10时,已经不算动态内存的开辟访问的空间范围内,是越界访问,
4.3 对非动态开辟内存使用free释放
void test (){int a = 10 ;int *p = &a;free (p); //ok?}
这个free只能对动态内存的空间释放,注意这一点
4.4 使用free释放一块动态开辟内存的一部分
void test (){int *p = ( int *) malloc ( 100 );p++;free (p); //p 不再指向动态内存的起始位置}
这个p指针发生改变,不在指向动态内存的起始位置,释放时只释放p现在指向的位置空间,所以只释放一部分,另一部分没释放,造成内存泄漏
4.5 对同一块动态内存多次释放
void test (){int *p = ( int *) malloc ( 100 );free (p);free (p); // 重复释放}
一个动态内存的开辟只能释放一次,不能多次释放
4.6 动态开辟内存忘记释放(内存泄漏)
void test (){int *p = ( int *) malloc ( 100 );if ( NULL != p){*p = 20 ;}}int main (){test();while ( 1 );}
这个test函数返回时,函数空间释放,所以找不到动态内存的的地址了,但动态内存空间还没释放,并且也释放不了,就成为内存泄露的问题
5. 动态内存经典笔试题分析
5.1 题目1:
void GetMemory ( char *p){p = ( char *) malloc ( 100 );}void Test ( void ){char *str = NULL ;GetMemory(str);strcpy (str, "hello world" );printf (str);}
当这个GetMemory函数返回时,函数空间释放,访问不到动态内存的空间了。但动态内存没释放,形成内存泄漏,由于形参是实参的临时拷贝,不影响str依旧是空指针,对空指针访问,程序崩溃
5.2 题目2:
char * GetMemory ( void ){char p[] = "hello world" ;return p;}void Test ( void ){char *str = NULL ;str = GetMemory();printf (str);}
首先注意这个GetMemory函数里是栈空间的变量数组,随着函数的释放,这个变量的空间也会释放,你虽然返回了数组首元素的地址,但是这个空间已经交还给系统,无权访问了,是野指针,所以我不确定到底能不能再次访问到这个数组,有可能还没有被系统把这个空间覆盖成其他内容,有可能访问到
5.3 题目3:
void GetMemory ( char **p, int num){*p = ( char *) malloc (num);}void Test ( void ){char *str = NULL ;GetMemory(&str, 100 );strcpy (str, "hello" );printf (str);}
这个是传str地址过去,是传址调用,那就用二级指针的形参接收,对二级指针解引用,将动态内存的首地址通过传址调用,让str接收到,所以此刻虽函数空间释放了,但我的动态内存的首地址拿到了,所以此刻这个str不是空指针了,可以strcpy,但可惜这个代码最终忘记释放str了,只有这一个小问题
5.4 题目4:
void Test ( void ){char *str = ( char *) malloc ( 100 );strcpy (str, "hello" );free (str);if (str != NULL ){strcpy (str, "world" );printf (str);}}
提早释放动态内存,但是只是对这个动态内存的空间没有访问的权限了,地址还是在的,通过strcpy,访问了动态内存的空间,这就是非法访问了,也就是说在没释放前,hello被拷贝过去,释放后,world无法拷贝过去
6. 柔性数组

struct st_type{int i;int a[ 0 ]; // 柔性数组成员};
有些编译器会报错无法编译可以改成:
struct st_type{int i;int a[]; // 柔性数组成员};
6.1 柔性数组的特点:
typedef struct st_type{int i;int a[ 0 ]; // 柔性数组成员}type_a;int main (){printf ( "%d\n" , sizeof (type_a)); // 输出的是 4return 0 ;}
6.2 柔性数组的使用
// 代码 1# include <stdio.h># include <stdlib.h>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 ;}
这样柔性数组成员a,相当于获得了100个整型元素的连续空间。

6.3 柔性数组的优势
// 代码 2# include <stdio.h># include <stdlib.h>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,所以你不能指望用户来发现这个事。所以,如果我们把结构体的内存以及其成员要的内存一次性分配好了,并返回给用户一个结构体指针,用户做一次free就可以把所有的内存也给释放掉。
连续的内存有益于提高访问速度,也有益于减少内存碎片。(其实,我个人觉得也没多高了,反正你跑不了要用做偏移量的加法来寻址)
结束语
动态内存的存储算是总结完了,动态内存我个人感觉也算是比较难,有点绕,可以多来回看看这篇博客,有什么问题跟我讨论,下一篇博客见
OK感谢观看!!!
相关文章:
C语言动态内存的管理
前言 本篇博客就来探讨一下动态内存,说到内存,我们以前开辟空间大小都是固定的,不能调整这个空间大小,于是就有动态内存,可以让我们自己选择开辟多少空间,更加方便,让我们一起来看看动态内存的有…...
CASIA数据集转png HWDB2.0-2.2
https://nlpr.ia.ac.cn/databases/handwriting/Home.html CASIA在线和离线中文手写数据库 https://nlpr.ia.ac.cn/databases/handwriting/Offline_database.html CASIA-HWDB2.0-2.2 离线文本数据库是由孤立字符数据集的作者制作的。每人撰写了五页给定文本。由于数据丢失&a…...
学习或复习电路的game推荐:nandgame(NAND与非门游戏)、Turing_Complete(图灵完备)
https://www.nandgame.com/ 免费 https://store.steampowered.com/app/1444480/Turing_Complete/ 收费,70元。据说可以导出 Verilog !...
前端面试题《react》
说说React render方法的原理?在什么时候会被触发? render函数里面可以编写JSX,转化成createElement这种形式,用于生成虚拟DOM,最终转化成真实DOM 在 React 中,类组件只要执行了 setState 方法,…...
快速入门Kotlin③类与对象
类 构造函数 主构造函数:主构造函数是类头的一部分,它跟在类名后面。主构造函数没有函数体,它可以包含初始化代码和属性声明。初始化块:init关键字修饰,它直接写在类体中。它的执行顺序与它们在类体中的出现顺序一致。 次构造函数:次要构造函数是可选的,用于提供额外…...
RUST:Arc (Atomic Reference Counted) 原子引用计数
在Rust编程语言中,Arc 是一个智能指针类型,全称为 "Atomic Reference Counted"(原子引用计数)。它的主要作用是提供线程安全的共享所有权机制,使得多个线程可以同时持有同一个数据结构的访问权,并…...
从0写一个问卷调查APP的第13天-1
1.今日任务 我也只是一个大学生,有什么思路不对的地方给我指出来哟! 分析:上次我们实现了任务调查的插入。但是我们插入的问卷调查只有它的标题,也就是这个问卷调查是什么我们告诉数据库了,但是现在我们还没有给它添加任何问题&…...
20.Python从入门到精通—参数 位置参数 关键字参数 默认参数 匿名函数 return 语句 强制位置参数
20.从入门到精通:参数 位置参数 关键字参数 默认参数 匿名函数 return 语句 强制位置参数 参数位置参数关键字参数默认参数 匿名函数return 语句强制位置参数 参数 在Python中,函数可以接受任意数量的参数,包括位置参数、关键字参数和默认参数。以下是这…...
Python爬虫之requests库
1、准备工作 pip install requests 2、实例 urllib库中的urlopen方法实际上就是以GET方式请求网页,requests库中相应的方法就是get方法。 import requestsr requests.get(https://www.baidu.com/) print(type(r)) # <class requests.models.Response> 响…...
鱼塘钓鱼(多路归并)
有 N 个鱼塘排成一排,每个鱼塘中有一定数量的鱼,例如:N5 时,如下表: 鱼塘编号12345第1分钟能钓到的鱼的数量 (1…1000)101420169每钓鱼1分钟钓鱼数的减少量(1…100)24653当前鱼塘到…...
java每日一题——买啤酒(递归经典问题)
前言: 非常喜欢的一道题,经典中的经典。打好基础,daydayup!!!啤酒问题:一瓶啤酒2元,4个盖子可以换一瓶,2个空瓶可以换一瓶,请问10元可以喝几瓶 题目如下: 啤酒问题:一瓶…...
最近接到一个大项目,给公司设计抢商品代金劵业务
我们公司是做汽车金融方面的工作,在业内还挺大。目前单量来源于2,3线城市,随着大环境越老越差位了吸引他们, 公司决定给全国的客户,销售等发一些商品 1.总体学习了京东开源秒杀系统设计思路和方案。 我们公司决定进行…...
防火墙(讲解)
目录 1.防火墙是什么? 2.防火墙设备 3.防火墙功能 1)出色的控制能力,过滤掉不安全服务 2)过滤非法用户和访问特殊站点 3)它能够对网络存取和访问进行监控审计 4.防火墙的局限 (1)防火墙有可能是可以绕过的 (2)防火墙不能防止内部出卖性攻击或者内部误操作…...
Python之装饰器-带参装饰器
Python之装饰器-带参装饰器 带参装饰器 之后不是一个单独的标识符,是一个函数调用函数调用的返回值又是一个函数,此函数是一个无参装饰器带参装饰器,可以有任意个参数 func()func(1)func(1, 2) def add(x, y):"""函数说明&…...
抖音IP属地怎么更改
抖音是一个非常受欢迎的短视频平台,吸引了无数用户在上面分享自己的生活和才艺。然而,随着快手的火爆,一些用户开始担心自己的IP地址会被他人获取,引起个人隐私风险。那么,抖音用户又该如何更改到别的地方呢࿱…...
Flutter 全局控制底部导航栏和自定义导航栏的方法
1. 介绍 导航栏在移动应用中扮演着至关重要的角色,它是用户与应用之间进行导航和交互的核心组件之一。无论是简单的页面切换,还是复杂的应用导航,导航栏都能够帮助用户快速找到所需内容,提升用户体验和应用的易用性。 在移动应用…...
检索增强生成(RAG)技术:实现流程、作用及应用案例
一. RAG简介 在自然语言处理(NLP)领域中,检索增强生成(Retrieval-Augmented Generation, RAG)技术巧妙地结合了信息检索与神经网络生成模型的力量,通过在生成过程中引入相关的外部信息,实现了在…...
Ubuntu安装和使用
Ubuntu 安装和配置 修改下载源 打开软件与更新, 选择其它站点, 选择中国, 选择阿里云源 谷歌中文输入法配置 Ctrl Alt T打开终端, 执行下述命令下fcitx框架 输入密码进行安装 sudo apt-get install -y fcitx-googlepinyinWin呼出菜单, 选择语言支持, 第一次打开会显示语言…...
【Unity】Stream最好用的Selfhost开源轻量服务
【背景】 有好几种场景的投屏或者远控应用希望实现,无论用哪种方式,都绕不开如何构建服务这一关。 【分析】 外网有很多直接付费使用的信令传输类型或是提供流服务的服务器,但我的目标场景是断绝外网的局域网,而且付费也总觉得…...
Web 常见的攻击方式有哪些?
常见的 Web 攻击方式有以下几种: 跨站脚本攻击(XSS 攻击) 跨站请求伪造(XSRF 攻击) SQL 注入 XSS 攻击 MDN 定义如下: 跨站脚本攻击(Cross-site scripting,XSS)是一…...
业务系统对接大模型的基础方案:架构设计与关键步骤
业务系统对接大模型:架构设计与关键步骤 在当今数字化转型的浪潮中,大语言模型(LLM)已成为企业提升业务效率和创新能力的关键技术之一。将大模型集成到业务系统中,不仅可以优化用户体验,还能为业务决策提供…...
【网络】每天掌握一个Linux命令 - iftop
在Linux系统中,iftop是网络管理的得力助手,能实时监控网络流量、连接情况等,帮助排查网络异常。接下来从多方面详细介绍它。 目录 【网络】每天掌握一个Linux命令 - iftop工具概述安装方式核心功能基础用法进阶操作实战案例面试题场景生产场景…...
R语言AI模型部署方案:精准离线运行详解
R语言AI模型部署方案:精准离线运行详解 一、项目概述 本文将构建一个完整的R语言AI部署解决方案,实现鸢尾花分类模型的训练、保存、离线部署和预测功能。核心特点: 100%离线运行能力自包含环境依赖生产级错误处理跨平台兼容性模型版本管理# 文件结构说明 Iris_AI_Deployme…...
无法与IP建立连接,未能下载VSCode服务器
如题,在远程连接服务器的时候突然遇到了这个提示。 查阅了一圈,发现是VSCode版本自动更新惹的祸!!! 在VSCode的帮助->关于这里发现前几天VSCode自动更新了,我的版本号变成了1.100.3 才导致了远程连接出…...
css的定位(position)详解:相对定位 绝对定位 固定定位
在 CSS 中,元素的定位通过 position 属性控制,共有 5 种定位模式:static(静态定位)、relative(相对定位)、absolute(绝对定位)、fixed(固定定位)和…...
NFT模式:数字资产确权与链游经济系统构建
NFT模式:数字资产确权与链游经济系统构建 ——从技术架构到可持续生态的范式革命 一、确权技术革新:构建可信数字资产基石 1. 区块链底层架构的进化 跨链互操作协议:基于LayerZero协议实现以太坊、Solana等公链资产互通,通过零知…...
人工智能(大型语言模型 LLMs)对不同学科的影响以及由此产生的新学习方式
今天是关于AI如何在教学中增强学生的学习体验,我把重要信息标红了。人文学科的价值被低估了 ⬇️ 转型与必要性 人工智能正在深刻地改变教育,这并非炒作,而是已经发生的巨大变革。教育机构和教育者不能忽视它,试图简单地禁止学生使…...
08. C#入门系列【类的基本概念】:开启编程世界的奇妙冒险
C#入门系列【类的基本概念】:开启编程世界的奇妙冒险 嘿,各位编程小白探险家!欢迎来到 C# 的奇幻大陆!今天咱们要深入探索这片大陆上至关重要的 “建筑”—— 类!别害怕,跟着我,保准让你轻松搞…...
【MATLAB代码】基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),附源代码|订阅专栏后可直接查看
文章所述的代码实现了基于最大相关熵准则(MCC)的三维鲁棒卡尔曼滤波算法(MCC-KF),针对传感器观测数据中存在的脉冲型异常噪声问题,通过非线性加权机制提升滤波器的抗干扰能力。代码通过对比传统KF与MCC-KF在含异常值场景下的表现,验证了后者在状态估计鲁棒性方面的显著优…...
关于easyexcel动态下拉选问题处理
前些日子突然碰到一个问题,说是客户的导入文件模版想支持部分导入内容的下拉选,于是我就找了easyexcel官网寻找解决方案,并没有找到合适的方案,没办法只能自己动手并分享出来,针对Java生成Excel下拉菜单时因选项过多导…...



