【C语言】动态内存管理,详细!!!

文章目录

  • 前言
  • 一、为什么存在动态内存分配
  • 二、动态内存开辟函数的介绍
    • 1.malloc
    • 2.calloc
    • 3.realloc
    • 4.free
  • 三、动态内存开辟中的常见错误
    • 1.误对NULL进行解引用操作
    • 2.对于动态开辟的空间进行了越界访问
    • 3.对于非动态开辟的内存进行了free操作
    • 4.只free掉动态开辟内存的一部分
    • 5.多次free已经释放的空间内存
  • 四、总结


添加链接描述

前言

大家好呀,时隔好几天小小樊又来为大家分享C语言学习啦,今天为大家分享一下自己对于动态内存管理的理解!!!

一、为什么存在动态内存分配

对于栈上开辟的空间:

  1. 空间开辟大小是固定的。
  2. 数组在申明的时候,必须指定数组的长度,它所需要的内存在编译时分配。但是对于空间的需求,不仅仅是上述的情况。有时候我们需要的空间大小在程序运行的时候才能知道,那数组的编译时开辟空间的方式就不能满足了

这时就需要在堆区矩形动态内存开辟了

二、动态内存开辟函数的介绍

1.malloc

大家先看一下库中对于他的说明:
在这里插入图片描述

函数功能:开辟内存块
参数size_t:需要申请的字节数
返回值:申请失败返回空指针,申请成功返回指向申请该空间首地址的指针头文件:stdlib.h
注意:返回指针的类型是void*,这时候需要你把该指针强制类型转化为你想要的类型,这样方便访问,以及解引用,malloc申请来的空间是连续的,但是多次malloc来的是不连续的
malloc的使用

int main()
{int*p=(int*) malloc(40);//申请了40个字节,强制转化为int*类型指针if (p == NULL)//如果返回空指针的话,申请失败{perror("malloc:");//打印错误信息return 1;//非正常退出}for (int i = 0; i < 10; i++){*(p + i) = i;//对每一个四个字节大小的元素赋值,这里*(p+i)的本质就是p[i];printf("%d", *(p + i));//打印每个元素}return 0;//程序正常退出}

2.calloc

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:申请一个数组在内存中,并且初始化为0;
参数:size_t num申请数组元素的个数,size_t size每个元素的字节大小
返回值:申请失败返回空指针,申请成功返回指向申请该空间首地址的指针头文件:stdlib.h
calloc函数的使用:

int main()
{int i = 0;int*p=(int*) calloc(10,sizeof(int));//申请10个元素,每个元素字节大小4if (p == NULL)//如果返回空指针的话,申请失败{perror("calloc:");//打印错误信息return 1;//非正常退出}for (int i = 0; i < 10; i++){printf("%d ", *(p + i));//打印初始化的值}free(p);p = NULL;return 0;}

在这里插入图片描述
malloc和calloc的区别:
与函数 malloc 的区别只在于 calloc 会在返回地址之前把申请的空间的每个字节初始化为全0

3.realloc

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:内存块的扩容
参数:第一个参数接收要扩容内存块的首地址,扩容后总字节大小(包括原来的字节大小)
头文件:stdlib.h
初始扩容的空间为空,则realloc和malloc的用法一模一样
返回值:扩容后空间的首地址
在这里插入图片描述
在这里插入图片描述
realloc函数的使用:

int main()
{int* p = (int*)malloc(40);//申请了40个字节,强制转化为int*类型指针if (p == NULL)//如果返回空指针的话,申请失败{perror("malloc:");//打印错误信息return 1;//非正常退出}for (int i = 0; i < 10; i++)//循环打印扩容前的元素{*(p + i) = i;printf("%d ", *(p + i));}int* ptr = (int*)realloc(p, 80);//原空间够用ptr==p,不够用的话ptr存放新地址if (ptr != NULL)//扩容成功{p = ptr;//原空间够用ptr==p,不够用的话ptr存放新地址,重新将新地址给p}for (int i = 10; i < 20; i++)//扩容后新空间的{*(p + i) = i;printf("%d ", *(p + i));}free(p);p = NULL;return 0;
}

在这里插入图片描述

4.free

大家先看一下库中对于他的说明:
在这里插入图片描述
功能:释放内存块
参数:指针接收要释放内存块的首地址
头文件:stdlib.h
返回值:无

注意:
当p所指向的申请的空间释放时,p指针指向随机位置,p变成野指针,所以我们要在释放后将其置为空!!!

如果我们不释放动态内存申请的内存的时候,程序结束,动态申请内存由操作系统自动回收,如果不用free函数释放申请好的空间,就会在程序运行结束前一直存在于堆中,造成内存泄漏

三、动态内存开辟中的常见错误

1.误对NULL进行解引用操作

比如:

int main()
{int* p = (int*)malloc(1000);int i = 0;if (p ==NULL){return 1;}for (i = 0; i < 250; i++){*(p + i) = i;}free(p);p = NULL;return 0;
}

当开辟内存失败时会返回空,这时容易造成此错误。
解决方法:开辟内存后进行判断,如上面代码中的 if 判断

2.对于动态开辟的空间进行了越界访问

int main()
{int* p = (int*)malloc(100);int i = 0;if (p ==NULL){return 1;}for (i = 0; i <=25; i++)//越界访问{*(p + i) = i;}free(p);p = NULL;return 0;
}

解决方法:人为进行检查是否越界

3.对于非动态开辟的内存进行了free操作

int main()
{int a = 10;int* p = &a;free(p);p = NULL;return 0;
}

4.只free掉动态开辟内存的一部分

int main()
{int* p = (int*)malloc(100);if (p == NULL){return 1;}int i = 0;for (i = 0; i < 10; i++){*p = i;p++;}free(p);p = NULL;return 0;
}

解决方案:
别改变保存动态开辟空间首地址的指针变量,使用时可以采用中间变量的方法!!!
例如:

int main()
{int* p = (int*)malloc(100);if (p == NULL){return 1;}int i = 0;for (i = 0; i < 10; i++){*(p+i)= i;printf("%d ", *(p + i));}free(p);p = NULL;return 0;
}

5.多次free已经释放的空间内存

int main()
{int* p = malloc(40);if (p == NULL){return 1;}free(p);free(p);p = NULL;return 0;
}

四、总结

本次内容到这里就分享完啦,如果大家觉得对自己有帮助的话还请大家点个赞呀,有分享不对的地方还恳请大家指正,谢谢大家的阅读!!!

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

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

相关文章

Blazor前后端框架Known-V1.2.13

V1.2.13 Known是基于C#和Blazor开发的前后端分离快速开发框架&#xff0c;开箱即用&#xff0c;跨平台&#xff0c;一处代码&#xff0c;多处运行。 Gitee&#xff1a; https://gitee.com/known/KnownGithub&#xff1a;https://github.com/known/Known 概述 基于C#和Blazo…

如何利用SFTP如何实现更安全的远程文件传输 ——【内网穿透】

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《高效编程技巧》《cpolar》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 文章目录 1. 安装openSSH1.1 安装SSH1.2 启动ssh 2. 安装cpolar2.1 配置termux服务 3. 远程SFTP连接配置3.1 查看生成的随机公…

redis实战-缓存三剑客穿透击穿雪崩解决方案

缓存穿透 定义 缓存穿透 &#xff1a;缓存穿透是指客户端请求的数据在缓存中和数据库中都不存在&#xff0c;这样缓存永远不会生效&#xff0c;这些请求都会打到数据库&#xff0c;造成数据库压力&#xff0c;也让缓存没有发挥出应有的作用 解决方案 缓存空对象 当我们客户端…

数字化转型能带来哪些价值?_光点科技

随着科技的迅猛发展&#xff0c;数字化转型已成为企业和组织的一项重要战略。它不仅改变了商业模式和运营方式&#xff0c;还为各行各业带来了诸多新的机遇和价值。在这篇文章中&#xff0c;我们将探讨数字化转型所能带来的价值。 数字化转型能够显著提升效率和生产力。通过引入…

redis常用五种数据类型详解

目录 前言&#xff1a; string 相关命令 内部编码 应用场景 hash 相关命令 内部编码 应用场景 list 相关命令 内部编码 应用场景 set 相关命令 内部编码 应用场景 Zset 相关命令 内部编码 应用场景 渐进式遍历 前言&#xff1a; redis有多种数据类型&…

多维时序 | MATLAB实现SCNGO-CNN-Attention多变量时间序列预测

多维时序 | MATLAB实现SCNGO-CNN-Attention多变量时间序列预测 目录 多维时序 | MATLAB实现SCNGO-CNN-Attention多变量时间序列预测预测效果基本介绍程序设计参考资料 预测效果 基本介绍 1.SCNGO-CNN-Attention超前24步多变量回归预测算法。 程序平台&#xff1a;无Attention适…

Docker容器学习:搭建私有镜像仓库Harbor

系统环境&#xff1a; Centos7.9Docker-ce:24 安装Docker-Compose curl -L "https://github.com/docker/compose/releases/download/v2.20.3/docker-compose-$(uname -s)-$(uname -m)" -o /usr/local/bin/docker-compose添加执行权限并验证 [rootnode4 ~]# chmod…

jenkins 日志输出显示时间戳的方式

网上很多方式比较片面&#xff0c;最新版插件直接使用即可无需更多操作。 使用方式如下&#xff1a; 1.安装插件 Timestamper 2.更新全局设置 系统设置-找到 Timestamper 勾选 Enabled for all Pipeline builds 也可修改时间戳格式。 帮助信息中显示 When checked, timesta…

单片机 (一) 让LED灯 亮

一&#xff1a;硬件电路图 二&#xff1a;软件代码 #include "reg52.h"#define LED_PORT P2void main() {LED_PORT 0x01; // 0000 0001 D1 是灭的 } #include "reg52.h" 这个头文件的作用&#xff1a;包含52 系列单片机内部所有的功能寄存器 三&#…

消息中间件介绍

消息队列已经逐渐成为企业IT系统内部通信的核心手段。它具有低耦合、可靠投递、广播、流量控制、最终一致性等一系列功能&#xff0c;成为异步RPC的主要手段之一。当今市面上有很多主流的消息中间件&#xff0c;如ActiveMQ、RabbitMQ&#xff0c;Kafka&#xff0c;还有阿里巴巴…

【Linux】线程篇Ⅱ:

线程Ⅱ &#x1f517;接上篇【线程篇Ⅰ】五、线程库 和 线程 id六、同步与互斥 &#x1f517;接上篇【线程篇Ⅰ】 &#x1f449;【Linux】线程篇Ⅰ&#xff1a;线程和task_struct 执行流的理解、相关接口命令、线程异常、线程的私有和共享 五、线程库 和 线程 id 对于 Linux …

Nginx高可用集群

目录 一.简介二.案例1.实现思路2.配置文件修改3.实现效果故障转移机制 一.简介 以提高应用系统的可靠性&#xff0c;尽可能地减少中断时间为目标&#xff0c;确保服务的连续性&#xff0c;达到高可用的容错效果。例如“故障切换”、“双机热备”、“多机热备”等都属于高可用集…