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.rhkb.cn/news/360356.html

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

相关文章

Python火焰锋动力学和浅水表面波浪偏微分方程

&#x1f3af;要点 &#x1f3af;流图可视化正弦余弦矢量场 | &#x1f3af;解空间变化边界条件二维拉普拉斯方程 | &#x1f3af;解圆柱坐标系标量场 | &#x1f3af;解一维泊松方程 | &#x1f3af;解二维扩散方程 | &#x1f3af;解火焰锋的动力学偏微分方程 | &#x1f3a…

[C#]winform基于opencvsharp实现黑白图像上色

【算法简介】 技术有时会提高艺术&#xff0c;但有时也会破坏艺术。着色黑白电影是一个可以追溯到1902年的古老想法。几十年来&#xff0c;许多电影创作者反对将黑白电影着色的想法&#xff0c;并将其视为对艺术的破坏。但今天它被接受为艺术形式的增强。该技术本身已经从艰苦…

2024年最新机动车签字授权人考试题库。

31."简易瞬态工况法"所使用的五气分析仪的温度范图:分析系统及相关部件应在&#xff08; &#xff09;。 A.0-40℃ B.0-50℃ C.0-60℃ D.-10-40℃ 答案:A 32.稀释氧传感器环境空气量程检测时的读数值位于&#xff08; &#xff09;%vol范围之外时&#xff0c;应…

[word] Word如何删除所有的空行? #职场发展#学习方法

Word如何删除所有的空行&#xff1f; 很多网友从网页复制文字粘贴到word文档后发现段落之间有空行&#xff0c;如果文字不多&#xff0c;手动删除这些空行也没有多少工作量&#xff0c;但是如果文字的字数达到成千上万&#xff0c;一个个手动删除这些空行还是很繁琐的。那么&a…

【Docker】Consul 和API

目录 一、Consul 1. 拉取镜像 2. 启动第一个consul服务&#xff1a;consul1 3. 查看consul service1 的ip地址 4. 启动第二个consul服务&#xff1a;consul2&#xff0c; 并加入consul1&#xff08;使用join命令&#xff09; 5. 启动第三个consul服务&#xff1a;consul3&…

解决ubuntu18.04 安装vscode 报依赖库错误,以及打不开终端的问题。

其实很简单&#xff0c;ubuntu18.04太老了&#xff0c;官网最新版本的vscode对ubuntu18.04会有些依赖库的问题。 一顿查资料后发现2023.11月的1.85版本正常使用&#xff0c;于是完美解决。 下载链接 Visual Studio Code November 2023 点击这里下载。 下载完成&#xff0c;…

iptables配置NAT实现端口转发

加载防火墙的内核模块 modprobe ip_tables modprobe ip_nat_ftp modprobe ip_conntrack 1.开启路由转发功能 echo net.ipv4.ip_forward 1 >> /etc/sysctl.conf sysctl -p2、将本地的端口转发到本机端口 将本机的 7777 端口转发到 6666 端口。 iptables -t nat -A PR…

MATLAB算法实战应用案例精讲-【数模应用】线性判别分析(附MATLAB、python和R语言代码实现)

目录 前言 算法原理 什么是判别分析 线性判别分析(LDA) 数学模型 二分类 多分类LDA ​编辑 算法思想: 费歇(FISHER)判别思想 贝叶斯(BAYES)判别思想 LDA算法流程 LDA与PCA对比 SPSSPRO 1、作用 2、输入输出描述 3、案例示例 4、案例数据 5、案例操作 …

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

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

【ai】tx2 nx: trition client安装nvidia-pyindex 一直失败

系统版本的pip和python虚拟环境的pipyolov4-triton-tensorrt的master分支 官方client jetson:pip3 install --user nvidia-pyindex 不成功啊 这个是让nvidia-pyindex 拉取nvidia@tx2-nx:~$ pip3 install --user nvidia-pyindex Collecting nvidia-pyindexDownloading https://…

Elasticsearch:倒数排序融合 - Reciprocal rank fusion - 8.14

警告&#xff1a;此功能处于技术预览阶段&#xff0c;可能会在未来版本中更改或删除。语法可能会在正式发布之前发生变化。Elastic 将努力修复任何问题&#xff0c;但技术预览中的功能不受官方正式发布功能的支持 SLA 约束。 倒数排序融合 (reciprocal rank fusion - RRF) 是一…

APP软件系统的开发流程

APP软件系统的开发是一个复杂的过程&#xff0c;需要多方面的知识和技能。建议选择专业的开发团队进行开发&#xff0c;以确保APP的质量和成功。APP软件系统的开发流程通常包括以下几个阶段。北京木奇移动技术有限公司&#xff0c;专业的软件外包开发公司&#xff0c;欢迎交流合…

Vue-条件渲染,事件绑定指令

条件渲染指令 条件渲染指令有两种&#xff1a; 两种指令大致相似 v-ifv-show 如果v-if的值为true&#xff0c;那么显示出内容&#xff0c;v-show也是一样 如果v-if的值为false&#xff0c;那么将不创建这个指令的标签&#xff0c;v-show将隐藏此标签 <body><div id&q…

最热门的智能猫砂盆好不好用?这期统统告诉你!

身为上班族的我们&#xff0c;常常被工作和出差填满日程。忘记给猫咪铲屎也不是一次两次了。但我们必须意识到&#xff0c;不及时清理猫砂盆不仅会让猫咪感到不适&#xff0c;还可能引发泌尿系统感染、皮肤疾病等健康问题。为了解决这个问题&#xff0c;越来越多的铲屎官开始将…

思科交换机基本配置命令

01进入特权模式enable switch>enable switch# 02进入全局配置模式configure terminal switch>enable switch#configure terminal switch(conf)# 03交换机命名hostname aptech2950以aptech2950为例 switch>enable switch#configure terminal switch(conf)#hostname apt…

驾校预约管理系统

摘 要 随着驾驶技术的普及和交通安全意识的增强&#xff0c;越来越多的人选择参加驾校培训&#xff0c;以获取驾驶执照。然而&#xff0c;驾校管理面临着日益增长的学员数量和繁琐的预约管理工作。为了提高驾校的管理效率和服务质量&#xff0c;驾校预约管理系统成为了必不可少…

C# Winform 开源UI库

WinForms&#xff0c;作为微软.NET框架下的一个桌面应用程序开发工具&#xff0c;自1999年首次亮相以来&#xff0c;已经走过了二十多年的发展历程。它以其简单直观的拖拽式界面设计和丰富的控件库&#xff0c;成为了大众喜爱的入门学习编程工具。由于它是比较基础的开发工具&a…

常见的结构型设计模式

设计模式&#xff08;二&#xff09; 常见的结构型模式 1.代理模式: 提供一种代理方法 &#xff0c;来控制对其他对象的访问。在有些情况下&#xff0c;一个对象不能或者不适合直接访问另一个对象&#xff0c;而代理对象可以在这两个类之间起一个中介的作用。 举例&#xf…

Charles抓取安卓应用https包演示

一、准备软件 夜神安卓模拟器 (yeshen.com) Charles (charlesproxy.com) 二、配置抓包 2.1 Charles安装PC根证书 记住这里的ip端口 三、安卓模拟器配置 3.1 配置安卓客户端网络代理 填写上文的ip端口&#xff0c;保存 3.2 安装根证书 3.2.1 导出根证书 linux主机执行 op…

《QT从基础到进阶·七十二》基于Qt开发的文件保险柜工具并支持文件各种加密和解密

1、概述 源码放在文章末尾 该项目实现了文件各种加密和解密的功能&#xff0c;能够有效的保障文件的安全&#xff0c;主要包含如下功能&#xff1a; 1、支持所有 AES 密钥长度&#xff1b; AES_128 AES_192 AES_256 2、支持ECB、CBC、CFB、OFB四种模式&#xff1b; 3、支持ZER…