C语言·动态内存管理

1. 为什么要有动态内存管理?

例1:

//固定的向内存申请4个字节
int a = 10;//申请连续的一块空间
int arr[10];

这些数据一旦声明定义之后就会在内存中有一块空间,这些空间都是固定的,为了让内存使用更加灵活,这时我们引入了动态内存分配

2. 动态内存分配的函数

使用这些函数之前,我们需要包含头文件stdlib,内存的申请都是在堆区上进行的

->1. malloc 函数

malloc向系统申请内存空间,申请到的空间没有初始化,直接返回的起始地址

开辟成功,则返回一个开辟好空间的指针

开辟失败,则返回一个NULL

若参数为0,nalloc的行为是标准未定义的,取决于编译器

void* malloc(size_t size);

(1). 需要开辟空间的总大小

例1:

#include <stdio.h>
#include <stdlib.h>int main()
{int*p = (int*)malloc(40);if(p == NUll){perror("malloc");return 1;}int i = 0;for(i = 0; i < 10; i++){*(p + i) = 0;}for(i = 0; i < 10; i++){printf("%d", p + i);} free(p);      p = NULL;
}

可以看到p向系统申请了10个字节的空间,在我们的操作下都赋值为了0

我们来看看将p释放之后:

将p释放了之后p还是指向的原地址,即:释放之后p变成了野指针

所以当我们释放了p之后,再将它手动置0是最安全的

->2. free函数

只能释放malloc ,calloc,realloc向内存申请的空间

如果free函数的参数为NULL,那么free函数不会进行任何操作

void free( void *memblock );

(1). void *memblock 需要释放的空间地址

例1:

以malloc为例:

#include <stdio.h>
#include <stdlib.h>int main()
{int*p = (int*)malloc(40);if(p == NUll){perror("malloc");return 1;}int i = 0;for(i = 0; i < 10; i++){*(p + i) = 0;}for(i = 0; i < 10; i++){printf("%d", p + i);} free(p);      p = NULL;
}

->3.calloc函数

在堆上申请空间(申请好空间后,会把空间初始化为0,然后返回起始地址)

void *calloc( size_t num, size_t size );

(1). size_t num 需要开辟空间的个数

(2). size_t size 每个的大小

例1:

#include <stdio.h>
#include <stdlib.h>int main()
{int *p = (int*)calloc(10, sizeof(int));if(p == NUll){perror("calloc");return 1;}int i = 0;for(; i < 10; i++){printf("%d", *(p + i));}free(p);p = NULL;return 0;
}

->4. realloc函数

有时我们会发现过去我们申请的内存小了,有时发现我们申请的内存大了为了应对这种情况,C语言引入了realloc函数来调整内存。

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

(1). void *memblock 要调整空间的地址

(2).size_t size 需要调整的空间的大小 

例1:

#include <stdio.h>
#include <stdlib.h>int main()
{int* p = (int*)malloc(sizeof(int) * 5);if(p == NULL){perror("malloc");return 1;}int i = 0;for(; i < 5; i++){*(p + i) = 1;}for(i = 0; i < 5; i++){printf("%d ", *(p + i));}printf("\n");int* ptr = (int*)realloc(p, sizeof(int) * 10);if(ptr == NULL){perror("realloc");return 1;}p = ptr;for(; i < 10; i++){*(p + i) = 1;}for(i = 0; i < 10; i++){printf("%d ", *(p + i));}free(p);p = NULL;return 0;
}

realloc的工作原理:

如例1所述:

->1. 后边有足够大的空间可以扩容时,realloc会直接在原有的基础上向后续上新的空间,返回旧的初始地址

->2. 后边没有足够大的空间可以扩容时,realloc函数会找一个满足空间大小的新的连续的空间,把旧的空间的数据拷贝到新空间的前面的位置,并且把旧的空间释放掉,同时返回新的空间的地址

->3. 如果realloc函数的参数是NULL,那么realloc函数的作用和malloc是一样的

例2:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>int main()
{int* p = (int*)realloc(NULL, 40);if(p == NULL){printf("%s", strerror(errno));return 1;}int i = 0;for(; i < 10; i++){*(p + i) = 1;}for(i = 0; i < 10; i++){printf("%d ", *(p + i));}free(p);p = NULL;return 0;
}

运行结果:

由图可知当realloc的参数是NULL时,它的作用和malloc是一样的

3. 常见的动态内存错误

->1. 对NULL解引用操作

malloc的返回值一定要判断

例1:

#include <stdio.h>
#include <stdlib.h>int main()
{int* p = (int*)malloc(40);if(p == NULL){perror("malloc");return 1;}//判断之后在使用return 0;
}

->2.对动态开辟内存的越界访问

越界访问,程序终端就挂了,卡死了

例1:

#include <stdio.h>
int main()
{int* p = (int*)malloc(100);if(p == NULL){perror("malloc");return 1;}int i = 0;for(i = 0; i < 100; i++){*(p + i) = 0;}free(p);p = NULL;return 0;
}

运行结果:

->3. 对非动态开辟内存使用free释放

使用free释放一块动态开辟内存的一部分

例1:

#include <stdio.h>
#include <stdlib.h>int main()
{int* p = (int*)malloc(100);if(p == NULL){perror("malloc");return 1;}int i = 0;for(i = 0; i < 25; i++){*p = i;p++;}free(p);p = NULL;return 0;
}

运行结果:

成因:

p指向的已经不再是这一百个字节的起始位置了或者指向这个空间的一部分,不是起始位置 

->5. 对同一块动态内存多次释放

例1:

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

运行结果:

程序直接挂掉

->6. 动态内存忘记释放(内存泄漏)

如果在函数中没有及时的释放动态内存,等函数销毁之后就没有机会了,只能等程序结束

例1:

#include <stdio.h>
#include <stdlib.h>void test()
{int* p = (int*)malloc(100);if(P == NULL){perror("malloc");return 1;}int i;for(i = 0; i < 25; i++){*(p + i) = 1;}for(i = 0; i < 25; i++){printf("%d ", *(p + i));} 
}int main()
{test();return 0;
}

解决方法:动态内存开辟函数应该和free函数成对使用

例2:

在函数中申请的内存没有使用完,将malloc开辟的空间的起始地址返回到main函数中继续使用,在使用完之后记得释放

#include <stdio.h>
#include <stdlib.h>int* test()
{int* p = (int*)malloc(100);if(p == NULL){perror("malloc");return 1;}return p;
}
int main()
{int* ptr = test();free(ptr);ptr = NULL;return 0;
}

4.为什么需要用free释放申请的内存

虽然我们不使用free释放空间到程序结束也会呗系统释放掉,但是我们如果碰到一直运行的程序呢!

while(1)
{malloc(1);    
}

他会一直吃掉系统的内存,导致系统的内存越来越少

用free释放空间也需要找准时机

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

这样的就是释放时机没把握好,该代码应该在if语句上面释放

5.例题

例1:

#include <stdio.h>
#include <stdlib.h>
#include <string.h>void GetMemory(char* p)
{p = (char*)malloc(100);
}void test(void)
{char* str = NULL;GetMemory(str);strcpy(str, "Hello World");printf(str);return 0;
}int main()
{test();return 0;
}

运行结果:

错误原因:

1.str传给p值的时候,p是str的一份临时拷贝,有自己独立的空间GetMemory在向系统申请空间之之后,放入了p中,在GetMemory返回之后,str的值依旧是NULL,即在strcpy拷贝时,形参非法范问了空间

2.在GetMemory函数申请空间之后,内存没有能及时的释放,造成了空间泄漏

将例1修改正确:

->1.第一种改法

#include <stdio.h>
#include <stdlib.h>
#include <string.h>void GetMemory(char** p)
{*p = (char*)malloc(100);
}void test(void)
{char* str = NULL;GetMemory(&str);strcpy(str, "Hello World");printf(str);free(str);str = NULL;return 0;
}int main()
{test();return 0;
}

运行结果:

->2.第二种改法

#include <stdio.h>
#include <stdlib.h>
#include <string.h>char* GetMemory()
{char* p = (char*)malloc(100);return p;
}void test(void)
{char* str = NULL;str = GetMemory();strcpy(str, "Hello World");printf(str);free(str);str = NULL;return 0;
}int main()
{test();return 0;
}

运行结果:

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

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

相关文章

【复旦邱锡鹏教授《神经网络与深度学习公开课》笔记】卷积

卷积经常用在信号处理中&#xff0c;用于计算信号的延迟累积。假设一个信号发射器每个时刻 t t t产生一个信号 x t x_t xt​&#xff0c;其信息的衰减率为 w k w_k wk​&#xff0c;即在 k − 1 k-1 k−1个时间步长后&#xff0c;信息为原来的 w k w_k wk​倍&#xff0c;时刻 …

SpringBoot开启事务日志

一般框架开启日志的方式&#xff1a; 开启某个包下的日志就写该包路径&#xff0c;开启某个类下的日志就写该类路径。

【数据结构】栈的定义与实现(附完整运行代码)

目录 一、栈的定义 二、顺序栈 链栈比较 三、栈的实现&#xff08;顺序栈&#xff09; 3.1 ❥ 定义栈结构 3.2 ❥ 初始化 3.3 ❥ 销毁 3.4 ❥ 插入&#xff08;入栈&#xff09; 3.5 ❥ 删除 &#xff08;出栈&#xff09; 3.6 ❥ 获取栈顶元素 3.7 ❥ 判空 3.8 ❥…

【Android】创建一个可以在屏幕上拖动的悬浮窗

项目需求 在界面上创建一个悬浮窗&#xff0c;可以自由的移动这个悬浮窗 需求解决 1.添加权限 <uses-permission android:name"android.permission.SYSTEM_ALERT_WINDOW"/>2.请求权限 从 Android 6.0 (API 23) 开始&#xff0c;应用需要动态请求显示悬浮窗…

F5《企业DNS建设白皮书》中的DNS解析服务器最佳实践

在这个数字化转型加速的时代&#xff0c;DNS&#xff08;域名系统&#xff09;的重要性不言而喻。每一次重大事件都凸显了DNS的可靠性和安全性问题。对企业而言&#xff0c;它不仅关系到业务连续性&#xff0c;更是提供永续数字服务的关键。本文根据F5公司发布的《企业DNS建设白…

中国4个民族群体的全基因组DNA甲基化变异图谱首次发布

2023年4月&#xff0c;由西北工业大学联合复旦大学等院校在Science China Life Sciences上发表题为“Genome-wide DNA methylation landscape of four Chinese populations and epigenetic variation linked to Tibetan high altitude adaptation”的文章&#xff0c;该研究通过…

【AI编译器】triton学习:编程模型

介绍 动机 在过去十年里&#xff0c;深度神经网络 (DNNs) 已成为机器学习 (ML) 模型的一个重要分支&#xff0c;能够实现跨领域多种应用中的最佳性能。这些模型由一系列包括参数化&#xff08;如滤波器&#xff09;和非参数化&#xff08;如缩小值函数&#xff09;元件组成的…

Android | 性能优化 之 TraceView工具的使用

上代码&#xff01; 先加权限&#xff1a; <uses-permission android:name"android.permission.WRITE_EXTERNAL_STORAGE"/> <uses-permission android:name"android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/> 选择跟踪范围,在开始追踪和结束…

景联文科技构建高质量多轮对话数据库,赋能AI交互新飞跃

近年来&#xff0c;大语言模型的发展极大推动了自然语言处理领域的进步&#xff0c;大语言模型正引领智能对话领域进入一个全新时代&#xff0c;不仅提升了对话体验的自然度和效率&#xff0c;也为探索更加人性化、智能化的交互方式开辟了道路。 景联文科技作为大语言模型数据服…

node.js 离线实时语音识别

前言 在node.js实现语音实时转文字。获取麦克风实时语音转文字。 下面是用vosk的效果。注意踩坑要及时评论哦&#xff0c;坑还是挺多的。 在探索后发现本地模型对设备还是有一定要求的&#xff0c;最总无奈采用百度语音识别的方案。 探索结果分享给大家&#xff0c;希望能在项…

AI视频教程下载-定制GPT:使用您的数据创建一个定制聊天GPT

Custom GPTs_ Create a Custom ChatGPT with Your Data 构建一个定制的GPT&#xff0c;与您自己的数据进行聊天。添加文档&#xff0c;生成图像&#xff0c;并集成API和Zapier。 这门全面的Udemy课程专为那些渴望学习如何创建自己定制版ChatGPT的人设计&#xff0c;以满足他们…

jstack的火焰图使用说明

1、jstack的官方文档说明 How to use Flame Graph? - Fast thread 2、jstack的文件分析网站&#xff0c;可以关注cpu消耗比较高的线程和火焰图 GC log analysis error

基于springboot+Vue高校宿舍管理系统的设计与实现【附源码】

本科毕业设计&#xff08;论文&#xff09; 基于springbootVue高校宿舍管理系统的设计与实现 目录 摘要 2 第一章 绪论 2 1.1 开发背景 2 1.2 开发意义 2 第二章 系统分析 3 2.1 系统的需求分析 3 2.2 系统开发设计思想 3 2.3系统开发步骤 3 2.4 系统的主要技术 4 2.4.1 B/S系…

JavaWeb系列八: WEB 开发通信协议(HTTP协议)

HTTP协议 官方文档什么是HTTP协议快速入门页面请求的一个问题(分析)http请求包分析(get)http请求包分析(post)GET请求 POST请求分别有哪些http响应包分析常用的状态码说明状态码200状态码404状态码500状态码302状态码304 MIME类型MIME介绍常见的 MIME 类型 官方文档 HTTP常见请…

七个值得收藏的资源网站,一定要码住~

1、壁纸网站&#xff1a;wallhere https://wallhere.com/ 这是一个免费的高清壁纸网站&#xff0c;各种类型的壁纸资源都有&#xff0c;高清无水印&#xff0c;每款壁纸都能下载到不同的尺寸&#xff0c;适应电脑、安卓手机和苹果手机的屏幕 2、电子书网站&#xff1a;熊猫搜…

java编写的界面可以调用python吗

如何使用Java调用Python程序 本文为大家介绍如何java调用python方法&#xff0c;供大家参考。 实际工程项目中可能会用到Java和python两种语言结合进行&#xff0c;这样就会涉及到一个问题&#xff0c;就是怎么用Java程序来调用已经写好的python脚本呢&#xff0c;一共有三种…

【源码+文档+调试讲解】牙科就诊管理系统

摘 要 现代经济快节奏发展以及不断完善升级的信息化技术&#xff0c;让传统数据信息的管理升级为软件存储&#xff0c;归纳&#xff0c;集中处理数据信息的管理方式。本牙科就诊管理系统就是在这样的大环境下诞生&#xff0c;其可以帮助管理者在短时间内处理完毕庞大的数据信息…

学法减分题库最新版,分享几个简单试用的学习和搜题工具 #微信#经验分享#知识分享

告别繁琐的查询步骤&#xff0c;用我们的拍照搜题功能&#xff0c;只需几秒钟&#xff0c;答案就出现在你眼前&#xff0c;让学习变得更加高效便捷。 1.减分侠 这是个辅助学分减分的公众号 根据新的学法减分考试大纲&#xff0c;涵盖小车、客车、货车、摩托车&#xff0c;各…

【AI落地应用实战】如何高效检索与阅读论文——302.AI学术论文工具评测

一、引言 作为一名学术领域的探索者&#xff0c;我们都知道&#xff0c;检索和阅读论文是我们获取知识、启发思考、验证假设的基石&#xff0c;也是日常学习中必不可少的基本功之一。然而在浩瀚的学术海洋中&#xff0c;如何快速、准确地找到我们需要的论文&#xff0c;就像是…

史上最全涵盖在线离线nginx安装手册(含国产信创环境下麒麟V10)

下载安装包略 下载地址&#xff1a;http://nginx.org/download/nginx-版本.tar.gz 配合下载资源食用更佳 https://download.csdn.net/download/ProGram_BlackCat/89480431 安装 tar -zxvf nginx-1.16.1.tar.gz && cd nginx-1.16.1# 创建安装目录(默认路径↓) mkdir /u…