C语言——动态内存管理

news/2024/4/27 12:06:16/文章来源:https://blog.csdn.net/Dirty_artist/article/details/129130992

目录

    • 0. 思维导图:
    • 1. 为什么存在动态内存分配
    • 2. 动态内存函数介绍
      • 2.1 malloc和free
      • 2.2 calloc
      • 2.3 realloc
    • 3. 常见的动态内存错误
      • 3.1 对NULL指针的解引用操作
      • 3.2 对动态内存开辟的空间越界访问
      • 3.3 对非动态开辟内存使用free释放
      • 3.4 使用free释放一块动态开辟内存的一部分
      • 3.5 对同一块动态内存多次释放
      • 3.6 动态内存开辟忘记释放(内存泄漏)
    • 4. C/C++程序的内存开辟

0. 思维导图:

在这里插入图片描述

1. 为什么存在动态内存分配

	int a = 10;//在栈空间开辟四个字节int arr[10] = { 0 };//在栈空间上开辟40个字节的连续空间

此类开辟空间的方式有两个特点:

  1. 空间开辟大小是固定的;
  2. 数组在申明的时候,必须指定数组长度,它所需要的内存在编译时分配(变长数组是不能改变数组的大小的,仅仅是允许数组的大小可用变量指定)。

但是对于空间大小的要求,有时候是需要在程序运行的时候才知道的,这时就需要使用动态内存开辟了。

2. 动态内存函数介绍

C语言分为3种内存池:栈区、堆区、静态区,而我们的动态内存函数,是属于堆区。
在这里插入图片描述

2.1 malloc和free

malloc参数及返回类型:
void* malloc (size_t size);

malloc可以向内存申请一块连续可用的空间,并返回指向这块空间的指针。

  • 如果开辟成功,则返回一个指向开辟好空间的指针。
  • 如果开辟失败,则返回一个NULL指针(所以在使用malloc时,一定要先检查返回值,看是否开辟成功)。
  • 返回值的类型是 *void,具体使用什么类型,由使用者来决定。
  • 如果size为0,malloc的行为是标准未定义的,取决于编译器。
    (该行为毫无意义,就好比找人借钱
    A:兄弟,最近手头有点紧,借点钱花花。
    B:借多少?
    A:借0元
    B:滚!)

因为malloc是在堆区上申请的内存空间,使用完毕之后需要将内存归还,所以C语言提供了内外一个free函数,专门用来做动态内存的释放和回收的。

free的参数及返回类型:
void free (void* ptr);

free函数用来释放动态开辟的内存。

  • 如果参数ptr指向的空间不是动态内存开辟的,那free函数的行为是未定义的。
  • 如果参数ptr是NULL指针,则函数什么事都不做。

malloc和free的声明都在stdlib.h头文件中,在使用时需引用头文件。
代码示例:

#include<stdlib.h>
#include<string.h>
#include<errno.h>
int main()
{//申请40个字节,用来存放10个整型int* ptr = (int*)malloc(40);if (ptr == NULL)//判断ptr是否申请成功{printf("%s\n", strerror(errno));return 1;}int i = 0;for (i = 0; i < 10; i++){*(ptr + i) = i + 1;printf("%d ", *(ptr + i));}//释放内存free(ptr);//如果不将ptr设置为空,则ptr将是野指针,所以需要我们主动置空ptr = NULL;return 0;
}

2.2 calloc

C语言还提供了一个函数叫calloccalloc函数也用来动态内存分配。

calloc的参数及返回类型:
void* calloc (size_t num, size_t size);

  • 函数的功能是为num个大小为size的元素开辟一块空间,并且把空间的每个字节初始化为0。
  • malloc的区别只在于calloc会在返回地址之前把申请的空间的每个字节初始化为0。

代码示例:

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

malloc申请到的空间,没有初始化,直接返回起始地址;
calloc申请到空间之后,会把空间初始为0,再返回起始地址。
如果申请的内存要求初始化,那么可用很方便使用calloc函数。
不过,因为malloc不需要初始化,所以整体来说malloc的效率会稍高于calloc
在这里插入图片描述

2.3 realloc

realloc函数的出现让动态内存的管理更加灵活,有时申请的空间大了,有事申请的空间又小了,那么为了合理的内存分配,就会使用realloc对动态内存就行调整。

realloc的参数及返回类型:
void* realloc (void* ptr, size_t size);

  • ptr是需调整的内存地址;
  • size调整之后的新大小;
  • 返回值为调整之后的起始地址;
  • 这个函数调整原内存空间大小的基础上,还好将原来内存中的数据移动到新的空间
  • realloc调整内存空间存在的两种情况:
    情况1:原有空间之后有足够大的空间
    情况2:原有空间之后没有足够大的空间
    在这里插入图片描述
    代码示例:
#include<stdlib.h>
#include<string.h>
#include<errno.h>
int main()
{int* p = (int*)malloc(5 * sizeof(int));if (p == NULL){perror("malloc");return 1;}int i = 0;for (i = 0; i < 5; i++){*(p + i) = 1;}//再向内存申请5个整型的空间//此时用新的指针地址接收,防止realloc申请失败,把原有的地址覆盖int* ptr = (int*)realloc(p, 10 * sizeof(int));if (ptr != NULL){p = ptr;}for (i = 5; i < 10; i++){*(p + i) = 1;}free(ptr);ptr = NULL;return 0;
}

3. 常见的动态内存错误

3.1 对NULL指针的解引用操作

int main()
{int*p = (int*)malloc(INT_MAX);//未对malloc的返回值进行判断int i = 0;for (i = 0; i < 10; i++){*(p + i) = 0;}free(p);p = NULL;return 0;
}

3.2 对动态内存开辟的空间越界访问

int main()
{int* p = (int*)malloc(100);//向内存申请了100个字节空间if (p = NULL){return 1;}int i = 0;for (i = 0; i < 100; i++){//此时访问的是100个整型的空间,应该需要400个字节//越界访问*(p + i) = 0;}free(p);p = NULL;return 0;
}

3.3 对非动态开辟内存使用free释放

int main()
{int a = 0;//栈区int* p = &a;free(p);p = NULL;
}

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

int main()
{int* p = (int*)malloc(100);if (p == NULL){return 1;}int i = 0;for (i = 0; i < 25; i++){*p = i;//p的地址发生改变p++;}free(p);//p未指向起始地址p = NULL;return 0;
}

3.5 对同一块动态内存多次释放

int main()
{int* p = (int*)malloc(100);if (p == NULL){return 1;}free(p);//...//将p释放,但未置空,此时p为野指针//如果将p释放后置空,那么在释放一次,free的参数为null,函数什么都不做free(p);return 0;
}

3.6 动态内存开辟忘记释放(内存泄漏)

void test()
{int* p = (int*)malloc(100);if (NULL != p){*p = 20;}
}
int main()
{test();while (1);
}

忘记释放不再使用的动态开辟的空间会造成内存泄漏,程序会一直吃内存(如下图)
在这里插入图片描述

使用mallocfree一定要成对使用。

4. C/C++程序的内存开辟

C/C++程序内存区域划分:
在这里插入图片描述
C/C++程序内存分配的几个区域:

  1. 栈区(stack):在执行函数时,函数内局部变量的存储单元都可以在栈上创建,函数执行结束时这些存储单元自动被释放。栈内存分配运算内置于处理器的指令集中,效率很高,但是分配的内存容量有限。 栈区主要存放运行函数而分配的局部变量、函数参数、返回数据、返回地址等。
  2. 堆区(heap):一般由程序员分配释放, 若程序员不释放,程序结束时可能由OS回收 。分配方式类似于链表。
  3. 数据段(静态区)(static)存放全局变量、静态数据。程序结束后由系统释放。
  4. 代码段:存放函数体(类成员函数和全局函数)的二进制代码。

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

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

相关文章

django+celery+ RabbitMQ自定义多个消息队列

关于django celery的使用网上有很多文章&#xff0c;本文就不多做更多的说明。 本文使用版本 python3.8.15 Django3.2.4 celery5.2.7celery.py from __future__ import absolute_import, unicode_literals import os from celery import Celery from kombu import Exchange, …

毕业后想从事软件测试,现在需要学习哪些内容呢

在你选择学习之前&#xff0c;要先考虑一下这个是不是你喜欢的发展方向&#xff0c;而不是只听别人推荐就直接做了选择先了解下软件测试是做什么的以及未来发展前景&#xff0c;最后才是如何自学 软件测试就是在测试这个软件是不是能够完全按照需求运行。软件测试岗再简单点说…

Windows启动docker客户端报错:Hardware assisted virtualization and enabled in the BIOS

报错内容 : &#x1f31f;1.在控制面板中点击 启用或关闭Windows功能&#x1f31f;2.勾选如下复选框&#x1f31f;3.Windows功能中没有Hyper-V复选框怎么办?(如果有请跳过此步骤)此时不同人的电脑还会出现没有Hyper-V选项的情况1.打开 Windows PowerShell&#xff0c;输入 sys…

如何效率搭建企业流程系统?试试低代码平台吧

编者按&#xff1a;本文介绍了一款可私有化部署的低代码平台&#xff0c;可用于搭建团队流程管理体系&#xff0c;并详细介绍了该平台可实现的流程管理功能。关键词:可视化设计&#xff0c;集成能力&#xff0c;流程审批&#xff0c;流程调试天翎是国内最早从事快速开发平台研发…

Hive内部表与外部表的区别具体说明

目录 1.在/opt/atguigu/目录下&#xff0c;新建两个txt文件 2.在hadoop的web端递归创建一个目录&#xff0c;存储这两个文件 3.查看web端的文件 一、内部表&#xff1a; 1.创建一个内部表&#xff0c;并指定内部表的存储位置 2.查看内部表&#xff0c;内部表中没有数据 …

2023.2 新方案 java代码混淆 java加密 字符串加密

Java字节码可以反编译&#xff0c;特别是创业公司,很好的项目很容易被别人破解反编译,造成很严重的损失,所以本混淆方案能很好的保护源码,而且在不断迭代,增强混淆效果,异常问题处理,达到保护项目的目的&#xff1a; 本次升级包括: 2023年02年19日 : ht-confusion-project-1.8…

PK体系下的教育场景—电子白板的应用

PK体系指基于国产飞腾&#xff08;Phytium&#xff09;CPU和麒麟&#xff08;Kylin&#xff09;操作系统的技术和产业体系&#xff0c;被誉为“中国架构”&#xff0c;目前基于PK体系的相关软硬件已经广泛用于党政、金融、电信等关基行业。教育信创在国家大战略布局下&#xff…

【技术分享】Web自动化之Selenium安装

Web 应用程序的验收测试常常涉及一些手工任务&#xff0c;例如打开一个浏览器&#xff0c;并执行一个测试用例中所描述的操作。但是手工执行的任务容易出现人为的错误&#xff0c;也比较费时间。因此&#xff0c;将这些任务自动化&#xff0c;就可以消除人为因素。Selenium 可以…

理解QPSK的实质-I右手正旋-Q左手负旋

正在学习5GNR PDCCH&#xff0c;用到QPSK。作一小结。 引言 我认为像我这样一个死民科&#xff0c;非主流非科班的通信人&#xff0c;理解QPSK的意义&#xff0c;甚至不比欧拉公式&#xff0c;或者是傅里叶变换小。 因为QPSK相较于BPSK&#xff0c;是真正第一次体现了调制的…

模拟默认密码自动生成-课后程序(JAVA基础案例教程-黑马程序员编著-第五章-课后作业)

【案例5-2】 模拟默认密码自动生成 【案例介绍】 1.任务描述 本例要求编写一个程序&#xff0c;模拟默认密码的自动生成策略&#xff0c;手动输入用户名&#xff0c;根据用户名自动生成默认密码。在生成密码时&#xff0c;将用户名反转即为默认的密码。 2.运行结果 运行结…

Power BI 数据处理介绍(数据初始调整、合并列及查看数据结构)

本系列的文章&#xff1a; 安装流程和示例介绍&#xff1a; 《Power BI windows下载安装流程&#xff09;》《Power BI 11个必学官方示例数据案例&#xff08;附下载链接&#xff09;》 数据导入阶段介绍&#xff1a; 《Power BI 数据导入&#xff08;SQL Server、MySQL、网页…

C++(42)-FSM-有限状态机

1.FSM 是什么&#xff1f; 一种用来进行对象行为建模的工具&#xff0c;用于描述对象在生命周期内所经历的状态序列&#xff0c;以及如何响应来自外界的各种事件。2.FSM 组成&#xff1a;状态、事件、动作3.FSM类型&#xff1a; 3.1Moore: 输出&#xff1a;当前状态有关…

mysql -学习总结

mysql 详解1、mysql特点2、事务2.1 事务的四大特性 – ACID2.2 并发事务问题2.3 事务的四大隔离级别2.4 事务隔离级别操作sql2.5 事务原理 – LBCC MVCC2.4.1 行的隐藏列2.4.2 ReadView2.4.3 MVCC在四种隔离级别下的区别2.5 undo log、binlog、redo log2.5.1 Undo log2.5.2 bin…

2023年2月22日PMP®项目管理认证课程正式开课

PMP认证是Project Management Institute在全球范围内推出的针对评价个人项目管理知识能力的资格认证体系。国内众多企业已把PMP认证定为项目经理人必须取得的重要资质。 PMP认证是Project Management Institute在全球范围内推出的针对评价个人项目管理知识能力的资格认证体系。…

安装MQTT Server遇到报错“cannot verify mosquitto.org‘s certificate”,该如何解决?

MQTT是基于发布/订阅的轻量级即时通讯协议&#xff0c;很适合用于低带宽、不稳定的网络中进行远程传感器和控制设备通讯等操作中。在我们的软件研发中&#xff0c;也经常使用MQTT协议进行消息通信等。今天来和大家分享一些关于在安装MQTT Server中遇到的疑难问题及解决思路。当…

文献综述怎么写?有哪些准备工作和内容要求

文献综述的撰写是提高研究生论文写作能力的重要途径&#xff0c;是研究生在撰写学术论文和学位论文中必须要涉及的内容&#xff0c;是不可或缺的&#xff0c;写好一篇好的文献综述是存在诸多困难和挑战的&#xff0c;需要掌握一定的技巧和方法。 一、文献综述的写作目的 文献综…

mysql常用且易混淆函数整理

DATE_FORMAT(date&#xff0c;format) 函数中format的格式如下&#xff1a; 类型转化函数 为了进行数据类型转化&#xff0c;MySQL提供了CAST()函数&#xff0c;它可以把一个值转化为指定的数据类型。类型有&#xff1a;BINARY,CHAR,DATE,TIME,DATETIME,SIGNED,UNSIGNED 示例&a…

Python|每日一练|数组|回溯|栈|树|双指针|单选记录:N 皇后|二叉树的前序遍历|四数之和

1、N 皇后&#xff08;数组&#xff0c;回溯&#xff09; n 皇后问题 研究的是如何将 n 个皇后放置在 nn 的棋盘上&#xff0c;并且使皇后彼此之间不能相互攻击。 给你一个整数 n &#xff0c;返回所有不同的 n 皇后问题 的解决方案。 每一种解法包含一个不同的 n 皇后问题 …

操作系统真相还原_第6章:完善内核

文章目录6.1 函数调用约定简介6.2 汇编语言和C语言混合编程汇编调用CC调用汇编6.3 实现打印函数流程程序编译并写入硬盘执行6.4 内联汇编简介汇编语言AT&T语法基本内联汇编扩展内联汇编6.1 函数调用约定简介 调用约定&#xff1a; calling conventions 调用函数时的一套约…

「mysql是怎样运行的」第5章 盛放记录的大盒子---InnoDB数据页结构

「mysql是怎样运行的」第五章 盛放记录的大盒子—InnoDB数据页结构 文章目录「mysql是怎样运行的」第五章 盛放记录的大盒子---InnoDB数据页结构[toc]一、不同类型的页介绍二、数据页结构的快速浏览三、记录在页中的存储记录头信息的秘密四、Page Directory(页目录)五、Page He…