当前位置: 首页 > news >正文

C语言——动态内存管理

目录

    • 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. 代码段:存放函数体(类成员函数和全局函数)的二进制代码。

相关文章:

C语言——动态内存管理

目录0. 思维导图&#xff1a;1. 为什么存在动态内存分配2. 动态内存函数介绍2.1 malloc和free2.2 calloc2.3 realloc3. 常见的动态内存错误3.1 对NULL指针的解引用操作3.2 对动态内存开辟的空间越界访问3.3 对非动态开辟内存使用free释放3.4 使用free释放一块动态开辟内存的一部…...

Docker安装Grafana

文章目录Grafana介绍拉取镜像准备相关挂载目录及文件启动容器访问测试添加 Prometheus 数据源常见问题看板配置Grafana介绍 上篇博客介绍了prometheus的安装&#xff1a; Docker部署Prometheus 在获取应用或基础设施运行状态、资源使用情况&#xff0c;以及服务运行状态等直观…...

数据结构(四):树、二叉树、二叉搜索树

数据结构&#xff08;四&#xff09;一、树1.树结构2.树的常用术语二、二叉树1.什么是二叉树2.二叉树的数据存储&#xff08;1&#xff09;使用数组存储&#xff08;2&#xff09;使用链表存储三、二叉搜索树1.这是什么东西2.封装二叉搜索树&#xff1a;结构搭建3. insert插入节…...

040、动态规划基本技巧(labuladong)

动态规划基本技巧 一、动态规划解题套路框架 基于labuladong的算法网站&#xff0c;动态规划解题套路框架&#xff1b; 1、基本介绍 基本套路框架&#xff1a; 动态规划问题的一般形式是求最值&#xff1b;核心如下&#xff1a; 穷举&#xff1b;明确base case&#xff1b;…...

html笔记(一)

一、html简介 什么是HTML&#xff1f; Hyper Text Markup Language 超文本标记语言 超文本&#xff1f;超级文本&#xff0c;例如流媒体&#xff0c;声音、视频、图片等。 标记语言&#xff1f;这种语言是由大量的标签组成。 任何一个标签都有开始标签和结束标签&…...

索引的情况

select * from A left join B on A.c B.c where A.employee_id 3 1.一句sql中 是可能走多次索引的&#xff0c;具体的 一般 表连接 &#xff0c;或者说生成临时表的时候&#xff0c;会走索引 然后条件过滤的时候也会走索引&#xff0c;具体的 还是要具体分析 2.表连接 字段…...

Verilog 学习第五节(串口发送部分)

小梅哥串口部分学习part1 串口通信发送原理串口通信发送的Verilog设计与调试串口发送应用之发送数据串口发送应用之采用状态机实现多字节数据发送串口通信发送原理 1&#xff1a;串口通信模块设计的目的是用来发送数据的&#xff0c;因此需要有一个数据输入端口 2&#xff1a;…...

破解遗留系统快速重构的5步心法(附实例)

前两天和一个架构师朋友闲聊&#xff0c;说到了 「重构」 这个话题&#xff0c;他们公司早年间上线的项目系统&#xff0c;因一直没专人在演进过程中为代码质量负责&#xff0c;导致现在代码越来越混乱&#xff0c;逐渐堆积成“屎山”&#xff0c;目前的维护成本已远高于重新开…...

信号量(上)实验

实验1&#xff1a;解决订票终端的临界区管理 订票终端是解决冲突问题&#xff0c;所以信号量的值是1 #include <stdio.h> #include <pthread.h> #include <unistd.h> #include <semaphore.h> int ticketAmout 2; // 票的数量: 全局变量 sem_t mutex…...

阿里5年,一个女工对软件测试的理解

成为一个优秀的测试工程师需要具备哪些知识和经验&#xff1f; 针对这个问题&#xff0c;可以直接拆分以下三个小问题来详细说明&#xff1a; 1、优秀软件测试工程师的标准是什么&#xff1f; 2、一个合格的测试工程师需要具备哪些专业知识&#xff1f; 3、一个合格的测试工程…...

前端练习项目

30 Web Projects 30 多个带有 HTML、CSS 和 JavaScript 的 Web 项目&#xff0c;由 Packt Publishing 提供 https://github.com/PacktPublishing/30-Web-Projects-with-HTML-CSS-and-JavaScript Small projects https://github.com/WebDevVikramChoudhary/small_projects_for_…...

sql复习(set运算符、高级子查询)

一、set运算符 union&#xff1a;得到两个查询结果的并集&#xff0c;并且⾃动去掉重复⾏。不会排序 union all&#xff1a;得到两个查询结果的并集&#xff0c;不会去掉重复⾏。也不会排序 intersect&#xff1a;得到两个查询结果的交集&#xff0c;并且按照结果集的第⼀个列进…...

整车电源的几种模式:OFF/ACC/RUN/CRANK

本文框架1.前言2. 四种电源模式2.1 OFF模式2.2 ACC模式2.3 ON模式2.4 CRANK模式3. KL15/KL301.前言 在诊断或者网络管理相关模块开发对客户的需求进行梳理时&#xff0c;经常会看到客户对不同车辆模式下处理策略的需求&#xff0c;如果前期没接触过这几种模式&#xff0c;可能…...

踩了大坑:wordpress后台 无法将上传的文件移动至wp-content

一、问题描述 今天迁移了wordpress站点至新服务器&#xff0c;结果上传图片出现“无法将上传的文件移动至wp-content/uploads”的提示&#xff0c;这是怎么回事&#xff0c;为什么会这样。 报错如下&#xff1a; 2023/02/20 08:57:48 [error] 9861#9861: *79624 FastCGI sen…...

page cache设计及实现

你好&#xff0c;我是安然无虞。 page cache的设计及实现 page cache 本质上也是一个哈希桶, 它是按照页的数量进行映射的. 当 central cache 向 page cache 申请内存时, page cache 先检查对应位置是否有span, 如果没有则向更大页去寻找一个span, 如果找到则分裂成两个. 比如…...

使用seata来解决分布式事务

文章目录 目录 文章目录 前言 一、Seata的执行流程如下 二、使用步骤 三、配置微服务客户端 总结 前言 Seata部署指南 Seata 是一款开源的分布式事务解决方案&#xff0c;致力于提供高性能和简单易用的分布式事务服务。Seata 将为用户提供了 AT、TCC、SAGA 和 XA 事务模…...

推荐一款新的自动化测试框架:DrissionPage

今天给大家推荐一款基于Python的网页自动化工具&#xff1a;DrissionPage。这款工具既能控制浏览器&#xff0c;也能收发数据包&#xff0c;甚至能把两者合而为一&#xff0c;简单来说&#xff1a;集合了WEB浏览器自动化的便利性和 requests 的高效率。 一、DrissionPage产生背…...

MQ系列面试

先来说说什么是MQ&#xff0c;MQ与多线程之间的区别MQ是消息中间件 可以实现异步 多线程也可以实现异步使用传统http协议方式调用接口存在的缺点如果服务器端没有及时的响应给客户端的时候&#xff0c;容易造成客户端阻塞等待。服务器响应超时 客户端发送重试机制 需要考虑避免…...

一句话设计模式2:原型模式

原型模式:每次得到一个新对象。 文章目录 原型模式:每次得到一个新对象。前言一、原型模式和new的区别二、如何实现原型模式1. 什么clone接口2. 开始使用,并验证浅clone效果3. 深度clone(也就是address也要复制一份)总结前言 原型模式可以说是目前接触的设计模式中,比较无用的…...

c++11特性与c++17特性

1、自动类型推导auto // C11 auto func1() -> int // 需要指定返回值类型 {return 10; }auto func2() -> std::function<void()> {auto lambda []() { };return lambda; }// c17 // 之后无需指定返回值类型 auto func1() {return 10; }auto func2() {auto lambda…...

无需配置环境!MinerU镜像一键部署,即刻体验智能文档解析

无需配置环境&#xff01;MinerU镜像一键部署&#xff0c;即刻体验智能文档解析 1. 为什么选择智能文档解析&#xff1f; 在日常办公和学习中&#xff0c;我们经常需要处理各种文档资料&#xff1a;PDF报告、扫描合同、学术论文、财务报表等。传统方式要么需要手动输入&#…...

从温控器到无人机:PID参数整定的‘手感’秘籍,附C语言代码避坑指南

从温控器到无人机&#xff1a;PID参数整定的‘手感’秘籍与实战避坑指南 在工业自动化和智能硬件开发中&#xff0c;PID控制算法就像一位隐形的调音师&#xff0c;默默调节着系统的每一个细微变化。无论是缓慢升温的工业烘箱&#xff0c;还是高速响应的四旋翼无人机&#xff0c…...

告别编译报错!手把手教你用Keil MDK5搭建GD32F103开发环境(含AC5编译器配置)

告别编译报错&#xff01;手把手教你用Keil MDK5搭建GD32F103开发环境&#xff08;含AC5编译器配置&#xff09; 嵌入式开发新手在初次接触GD32F103时&#xff0c;往往会被各种编译报错搞得焦头烂额。特别是从STM32转过来的开发者&#xff0c;本以为操作流程相似&#xff0c;结…...

ImageSearch:5分钟掌握本地千万级图片搜索的终极指南

ImageSearch&#xff1a;5分钟掌握本地千万级图片搜索的终极指南 【免费下载链接】ImageSearch 基于.NET8的本地硬盘千万级图库以图搜图案例Demo和图片exif信息移除小工具分享 项目地址: https://gitcode.com/gh_mirrors/im/ImageSearch 你是否曾在电脑里堆积如山的照片…...

Obsidian LaTeX Suite终极指南:让数学公式编辑如行云流水

Obsidian LaTeX Suite终极指南&#xff1a;让数学公式编辑如行云流水 【免费下载链接】obsidian-latex-suite Make typesetting LaTeX as fast as handwriting through snippets, text expansion, and editor enhancements 项目地址: https://gitcode.com/gh_mirrors/ob/obsi…...

Apache HBase与Spark集成终极指南:10个实时数据处理高效方案

Apache HBase与Spark集成终极指南&#xff1a;10个实时数据处理高效方案 【免费下载链接】hbase Apache HBase 项目地址: https://gitcode.com/GitHub_Trending/hb/hbase Apache HBase是一个高可靠性、高性能、面向列的分布式存储系统&#xff0c;非常适合存储海量结构化…...

ER-Save-Editor:开源工具实现艾尔登法环跨平台存档修改全指南

ER-Save-Editor&#xff1a;开源工具实现艾尔登法环跨平台存档修改全指南 【免费下载链接】ER-Save-Editor Elden Ring Save Editor. Compatible with PC and Playstation saves. 项目地址: https://gitcode.com/GitHub_Trending/er/ER-Save-Editor ER-Save-Editor作为一…...

单片机:从核心原理到智能应用实战

1. 单片机&#xff1a;智能时代的微型大脑 想象一下清晨醒来&#xff0c;窗帘自动拉开&#xff0c;咖啡机开始工作&#xff0c;室内温度始终保持在最舒适的状态——这些看似简单的智能场景背后&#xff0c;都藏着一个不起眼却至关重要的核心部件&#xff1a;单片机。这块比硬币…...

SQL 基础及 MySQL DBA 运维实战 - 6:Mycat代理技术

MySQL DBA运维实战&#xff1a;集群与代理技术深度解析 引言 在现代互联网应用中&#xff0c;数据库的高可用性、可扩展性和性能是企业级应用的核心需求。随着业务量的增长&#xff0c;单一数据库服务器往往无法满足需求&#xff0c;此时数据库集群和代理技术成为解决这些问题…...

AI Agent工程师进阶指南:掌握核心技能,冲击高薪(P7-P8必备)!

本文详细介绍了AI Agent工程师的能力分层&#xff0c;从API调用工程师到系统设计工程师再到基础设施架构师&#xff0c;明确了不同层级的能力要求和市场现状。文章深入剖析了核心技术栈&#xff0c;包括向量数据库、RAG系统、Agent架构、Memory系统以及生产化工程等关键领域&am…...