C语言——动态内存管理

news2025/2/27 16:37:39

目录

    • 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.coloradmin.cn/o/361376.html

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

相关文章

读书笔记//来自公众号(2)

非常喜欢阅读同行的文章&#xff0c;彷佛进行一场隔空交流。大家都是数据分析师&#xff0c;有许多共鸣&#xff1b;了解数据分析在不同行业的应用&#xff0c;往往很有收获。 这位朋友在零售行业、工业物联网、汽车互联网、2G电商等做个数据分析&#xff0c;有10多工作经验。…

【STC8A8K64D4开发板】——搭建开发环境

第1-2讲&#xff1a;搭建开发环境 需要安装的工具软件 表1&#xff1a;需要的工具软件 序号 软件工具 说明 1 Keil C51 9.52安装文件 Keil C51集成开发环境。 2 ch341ser驱动 USB最新驱动可到沁恒官网下载&#xff1a;www.wch.cn/downloads/CH341SER_EXE.html 3 st…

JVM运行时数据区

在上篇文章中我们详细介绍了JVM类加载子系统&#xff1a;https://blog.csdn.net/u011837804/article/details/129049418&#xff0c;Class文件被类加载子系统加载后就进入到了运行时数据区等待执行引擎执行&#xff0c;运行时数据区也是JVM中最重要的一部分&#xff0c;GC、OOM…

基于springboot+mybatis+mysql+vue运动会报名管理系统

基于springbootmybatismysqlvue运动会报名管理系统一、系统介绍二、功能展示1.用户登陆2.报名详情(运行员)3.比赛报名&#xff08;运动员&#xff09;4.个人参赛项目&#xff08;运动员&#xff09;5.加油稿展示&#xff08;运动员&#xff09;6.学院积分排名&#xff08;运动员…

Spring MVC 源码 - HandlerMapping 组件(三)之 AbstractHandlerMethodMapping

HandlerMapping 组件HandlerMapping 组件&#xff0c;请求的处理器匹配器&#xff0c;负责为请求找到合适的 HandlerExecutionChain 处理器执行链&#xff0c;包含处理器&#xff08;handler&#xff09;和拦截器们&#xff08;interceptors&#xff09;handler 处理器是 Objec…

K_A12_022 基于STM32等单片机驱动VL53L0X模块 串口与OLED0.96双显示

K_A12_022 基于STM32等单片机驱动VL53L0X模块 串口与OLED0.96双显示一、资源说明二、基本参数参数引脚说明三、驱动说明UART对应程序:IIC对应程序:四、部分代码说明1、接线引脚定义1.1、STC89C52RCVL53L0X模块1.2、STM32F103C8T6VL53L0X模块五、基础知识学习与相关资料下载六、…

打印流、转换流、数据流 、随机访问流

Java知识点总结&#xff1a;想看的可以从这里进入 目录5、打印流6、转换流7、数据流8、随机访问流5、打印流 实现将基本数据类型的数据格式转化为字符串输出&#xff0c;它们提供了一系列重载的print()和println()方法&#xff0c;用于多种数据类型的输出&#xff0c;这种流不会…

基于react+typescript的前端组件库violet-design(字节青训营项目)

文章目录前言一、项目介绍&#x1f48c; 介绍特性兼容性&#x1f4e6; 安装使用 npm 安装使用 yarn 安装浏览器引入&#x1f528; 示例1. 引入样式2. 使用组件按需加载TypeScript✨ 组件&#x1f517; 链接二、项目实现2.1 技术选型2.2 架构设计2.2.1 目标用户和场景2.2.2 组件…

容器技术概述

容器化应用程序 软件应用程序通常依赖于运行时环境提供的其他库、配置文件或服务。软件应用程序的传统运行环境是物理主机或虚拟机&#xff0c;应用程序依赖项作为主机的一部分安装。 例如&#xff0c;考虑一个 Python 应用程序&#xff0c;它需要访问实现 TLS 协议的公共共享…

第 16 章_多版本并发控制

第 16 章_多版本并发控制 1. 什么是MVCC MVCC &#xff08;Multiversion Concurrency Control&#xff09;&#xff0c;多版本并发控制。顾名思义&#xff0c;MVCC 是通过数据行的多个版本管理来实现数据库的并发控制。这项技术使得在InnoDB的事务隔离级别下执行一致性读操作…

剑指 Offer 27. 二叉树的镜像

剑指 Offer 27. 二叉树的镜像 难度&#xff1a;easy\color{Green}{easy}easy 题目描述 请完成一个函数&#xff0c;输入一个二叉树&#xff0c;该函数输出它的镜像。 例如输入&#xff1a; 镜像输出&#xff1a; 示例 1&#xff1a; 输入&#xff1a;root [4,2,7,1,3,…

Office 365 备份与恢复

Microsoft Office 365中的不同服务几乎可以随时访问&#xff0c;这要归功于Microsoft的99.9%正常运行时间记录。但是&#xff0c;Office 365步履蹒跚的一个方面是提供了一种从意外数据丢失中恢复的方法。Microsoft 提供的数据保留功能并非适用于所有数据丢失情况的可行解决方案…

亿级高并发电商项目-- 实战篇 --万达商城项目 十二(编写用户服务、发送短信功能、发送注册验证码功能、手机号验证码登录功能、单点登录等模块)

&#x1f44f;作者简介&#xff1a;大家好&#xff0c;我是小童&#xff0c;Java开发工程师&#xff0c;CSDN博客博主&#xff0c;Java领域新星创作者 &#x1f4d5;系列专栏&#xff1a;前端、Java、Java中间件大全、微信小程序、微信支付、若依框架、Spring全家桶 &#x1f4…

Leetcode力扣秋招刷题路-0082

从0开始的秋招刷题路&#xff0c;记录下所刷每道题的题解&#xff0c;帮助自己回顾总结 82. 删除排序链表中的重复元素 II 给定一个已排序的链表的头 head &#xff0c; 删除原始链表中所有重复数字的节点&#xff0c;只留下不同的数字 。返回 已排序的链表 。 示例 1&#…

这6个视频剪辑素材库,你一定要知道~

推荐5个免费商用视频素材网站&#xff0c;建议收藏哦&#xff01; 1、菜鸟图库 视频素材下载_mp4视频大全 - 菜鸟图库 网站素材量很大&#xff0c;有设计、图片、音频、视频等超多素材&#xff0c;大部分都能免费下载。视频素材都很高清&#xff0c;有自然、人物、科技、农业…

前端页面开发模块组织结构

模块组织 任何超过 1000 行的 CSS 代码,你都曾经历过这样的体验: 这个 class 到底是什么意思呢?这个 class 在哪里被使用呢?如果我创建一个 xxoo class,会造成冲突吗?Reasonable System for CSS Stylesheet Structure 的目标就是解决以上问题,它不是一个框架,而是通过…

Freemarker快速入门

freemarker提供很多指令用于解析各种类型的数据模型参考地址&#xff1a;http://freemarker.foofun.cn/ref_directives.html一.测试搭建Freemarker的运行环境并进行测试.1.添加Freemarker与SpringBoot的整合包XML <!-- Spring Boot 对结果视图 Freemarker 集成 --> <d…

互斥锁原理

如果有交互的公共数据区域&#xff0c;我们需要让一个进程先执行&#xff0c;一个进程后执行&#xff0c;互斥锁就是用锁的方式让他们的竞争关系变得有序。 临界区问题 临界区是在程序之间有公共数据交互时产生的区域&#xff0c;没有两个进程可以在它们各自的临界区同时执行…

我的 System Verilog 学习记录(1)

引言 技多不压身&#xff0c;准备开始学一些 System Verilog 的东西&#xff0c;充实一下自己&#xff0c;这个专栏的博客就记录学习、找资源的一个过程&#xff0c;希望可以给后来者一些借鉴吧&#xff0c;IC找工作的都加把油&#xff01; 本文是准备先简单介绍一下环境搭建…

C++11智能指针std::shared_ptr介绍及使用

介绍 shared_ptr是一种智能指针(smart pointer)&#xff0c;作用有如同指针&#xff0c;但会记录有多少个shared_ptrs共同指向一个对象。这便是所谓的引用计数(reference counting),比如我们把只能指针赋值给另外一个对象,那么对象多了一个智能指针指向它,所以这个时候引用计数…