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)是一…...
调用支付宝接口响应40004 SYSTEM_ERROR问题排查
在对接支付宝API的时候,遇到了一些问题,记录一下排查过程。 Body:{"datadigital_fincloud_generalsaas_face_certify_initialize_response":{"msg":"Business Failed","code":"40004","sub_msg…...
基于FPGA的PID算法学习———实现PID比例控制算法
基于FPGA的PID算法学习 前言一、PID算法分析二、PID仿真分析1. PID代码2.PI代码3.P代码4.顶层5.测试文件6.仿真波形 总结 前言 学习内容:参考网站: PID算法控制 PID即:Proportional(比例)、Integral(积分&…...
【机器视觉】单目测距——运动结构恢复
ps:图是随便找的,为了凑个封面 前言 在前面对光流法进行进一步改进,希望将2D光流推广至3D场景流时,发现2D转3D过程中存在尺度歧义问题,需要补全摄像头拍摄图像中缺失的深度信息,否则解空间不收敛…...
如何为服务器生成TLS证书
TLS(Transport Layer Security)证书是确保网络通信安全的重要手段,它通过加密技术保护传输的数据不被窃听和篡改。在服务器上配置TLS证书,可以使用户通过HTTPS协议安全地访问您的网站。本文将详细介绍如何在服务器上生成一个TLS证…...
Neo4j 集群管理:原理、技术与最佳实践深度解析
Neo4j 的集群技术是其企业级高可用性、可扩展性和容错能力的核心。通过深入分析官方文档,本文将系统阐述其集群管理的核心原理、关键技术、实用技巧和行业最佳实践。 Neo4j 的 Causal Clustering 架构提供了一个强大而灵活的基石,用于构建高可用、可扩展且一致的图数据库服务…...
iOS性能调优实战:借助克魔(KeyMob)与常用工具深度洞察App瓶颈
在日常iOS开发过程中,性能问题往往是最令人头疼的一类Bug。尤其是在App上线前的压测阶段或是处理用户反馈的高发期,开发者往往需要面对卡顿、崩溃、能耗异常、日志混乱等一系列问题。这些问题表面上看似偶发,但背后往往隐藏着系统资源调度不当…...
Linux 内存管理实战精讲:核心原理与面试常考点全解析
Linux 内存管理实战精讲:核心原理与面试常考点全解析 Linux 内核内存管理是系统设计中最复杂但也最核心的模块之一。它不仅支撑着虚拟内存机制、物理内存分配、进程隔离与资源复用,还直接决定系统运行的性能与稳定性。无论你是嵌入式开发者、内核调试工…...
基于SpringBoot在线拍卖系统的设计和实现
摘 要 随着社会的发展,社会的各行各业都在利用信息化时代的优势。计算机的优势和普及使得各种信息系统的开发成为必需。 在线拍卖系统,主要的模块包括管理员;首页、个人中心、用户管理、商品类型管理、拍卖商品管理、历史竞拍管理、竞拍订单…...
SQL慢可能是触发了ring buffer
简介 最近在进行 postgresql 性能排查的时候,发现 PG 在某一个时间并行执行的 SQL 变得特别慢。最后通过监控监观察到并行发起得时间 buffers_alloc 就急速上升,且低水位伴随在整个慢 SQL,一直是 buferIO 的等待事件,此时也没有其他会话的争抢。SQL 虽然不是高效 SQL ,但…...
MyBatis中关于缓存的理解
MyBatis缓存 MyBatis系统当中默认定义两级缓存:一级缓存、二级缓存 默认情况下,只有一级缓存开启(sqlSession级别的缓存)二级缓存需要手动开启配置,需要局域namespace级别的缓存 一级缓存(本地缓存&#…...



