浅谈 malloc 函数在单片机上的应用

2023-11-10

	聊聊 malloc函数 在单片机程序设计中怎么使用

前言

最近更新 RT-Thread 专栏到了内存管理的时候,想了想该怎么来说明这个内存管理,实际上在平时使用STM32做一般产品的时候基本不会用到 malloc 函数,即便是使用了操作系统,在业务逻辑不复杂的情况下,还是用不上malloc。

但是每个嵌入式 RTOS 都会有自己的内存管理方式,本文就来聊聊我对 malloc 函数在单片机程序设计中的一些看法。

本文并不是要说明在单片机中怎么使用 malloc函数,而是根据博主自己的理解,从函数使用的根本上来分析需不需要使用,何时何地使用。

本文的探讨是单片机领域,以 Cortex-M 系列内核为例。

一、malloc 函数简介

malloc的全称是memory allocation,中文叫动态内存分配。

函数原型 void *malloc(unsigned int size),专业解释还是套用百度百科:
在这里插入图片描述
对于malloc函数,应该所有嵌入式工程师都知道,即便没用过也都听过,通过上面简单的说明,也都能够知道是干什么用的。

注意上面红色框框部分,malloc开辟的是连续的空间,返回的是一个地址,当内存不使用,需要使用free()函数释放内存。

二、malloc 之于单片机

在我们的单片机程序设计中,大都使用的C语言,当然可以使用 malloc 函数,但是有很多人并不能够真正的理解它。

要理解单片机系统中的 malloc函数 ,首先必须了解动态分配的内存是在什么地方呢!

2.1 malloc 函数申请的内存在哪里?

也许大部分人知道在堆中!是的,在堆中没错。

那么接着问题,堆在单片机的什么地方呢?具体地址是多少呢?

要了解这个问题,就得了解内核的数据存储方式,我们以常用的STM32为例,我在下面这篇博文详细的介绍过STM32的内存管理(如果不懂的建议先看一下这篇博文):

STM32的内存管理相关(内存架构,内存管理,map文件分析)

先简单了解下单片机的堆栈,文中有如下说明:
在这里插入图片描述

如果通过上面推荐的博文理解了内存分配,那么我们就可以得到如下结论,可以知道 malloc 申请空间的准确地址了:
在这里插入图片描述

2.2 用与不用malloc的区别

知道了malloc申请空间在单片机中的地址,我们再来看一下用于不用的区别。

一般在我们的设计中,函数中可能会初始化一些临时变量,如果是一个数组,那么他也会申请一段内存空间,我们通过一张图来看看使用临时变量与malloc 的区别:
在这里插入图片描述

解释到这里,相信大家对在单片机上使用 malloc 有了一个更深的认识。他所存放的空间与我们经常局部变量的空间是不同,而且我们也知道了在什么位置。

至于单片机用还是不用 malloc函数? 别急,我们还得往下面分析分析。

三、malloc可能遇到的问题

还是官方的百度百科里面介绍malloc的工作机制时候,有下面的说明:
在这里插入图片描述
注意图中画红色的部分,简单解释就是,使用 malloc 函数多了以后,会产生很多的内存碎片,白白浪费内存。

3.1 内存碎片

什么是内存碎片是什么?

这种专业的术语还得靠万能的百度(虽然百度百科的解释针对的是大范围的,但是对于单片机来说其实是一样的):
在这里插入图片描述

系统中所有的不可用的空闲内存就是内存碎片。

那么 内存碎片是什如何产生的?

在上图中其实有碎片是如何产生的说明,内部碎片是因为处理器的体系结构,需要字节对齐,比如我们在单片机中,常有4字节对齐,8字节对齐,不要说这个也不知道,我随便打开一个 STM32L051 的启动文件说明(GCC环境下的链接文件):
在这里插入图片描述

按照这个启动文件,我们也应该能知道,堆栈内存空间是需要8字节对齐的,那么我们在STM32上使用 malloc 分配的内存空间是8字节对齐的,即便你用不上8个字节,系统也会给你对齐补上。

说到这个,正好了可以说明我们内存碎片产生的第一种情况,内部碎片的产生:
在这里插入图片描述
那么外部碎片的产生,我们也可以用图形来表示:
在这里插入图片描述
随着内存不断被分配和释放,整个内存区域会产生越来越多的碎片,因为在使用过程中,申请了一些内存,其中一些释放了,导致内存空间中存在一些小的内存块,它们地址不连续,不能够作为一整块的大内存分配出去,系统中还有足够的空闲内存,但因为它们地址并非连续,不能组成一块连续的完整内存块,会使得程序不能申请到大的内存。

在我们使用的单片机上,碎片产生问题尤为明显,平时接触不到,那是因为一般学习测试不会遇到复杂的项目。

3.2 内存管理

使用 malloc 会产生碎片,那么有什么办法可以解决这个问题?

当然是有的,那就是内存管理。

内存管理就是为了解决上面提到的内存碎片问题,如何高效,快速的分配,并且在适当的时候释放和回收内存资源。

对于单片机来说,如果你有能力是可以自己设计内存管理的方式。

如果使用嵌入式操作系统比如 FreeRTOS、RT-Thread 的话,他们内核是自带内存管理的,本文并不会讨论他们具体是如何内存管理的,但是有必要了解一下操作系统的思路。

以 FreeRTOS 为例子说明:

在 FreeRTOS 中有一个宏定义configTOTAL_HEAP_SIZE
在这里插入图片描述

操作系统首先向系统申请了一块大的内存,这块内存内存由操作系统自己的内存管理方式,对于FreeRTOS而言有5种内存管理方式:

在这里插入图片描述
我们在设计的时候可以自己选择使用哪一种,比如:
在这里插入图片描述

对于申请的这块内存由操作系统自动管理,FreeRTOS操作系统创建的任务,任务栈使用的就是这一块内存,同时使用pvPortMalloc函数申请动态内存,也会从这一块内存中分配,因为他有一套完善的内存管理方式,所以相对我们直接使用 malloc来说,他能够很好的处理系统内存碎片的问题。

既然说到这里,额外的一个问题,就是 FreeRTOS configTOTAL_HEAP_SIZE定义额这块内存处于单片机内存那个部分呢?

这就可以看我的又一篇博文:嵌入式RTOS的 任务栈 和 系统栈

FreeRTOS申请的内存是属于.bss段的,位置如下图所示:
在这里插入图片描述

对于单片机使用的嵌入式操作系统来说,他们有自己的内存管理方式,也会提供对于的动态内存申请结构,这时候我们使用操作系统提供的 malloc 接口函数,可以很好的避免内存碎片的产生。

注意!!单片机用了有内存管理的操作系统,系统会提供对于的 API,比如 FreeRTOS 的 pvPortMalloc 函数,RT-Thread 的 rt_malloc 函数。
如果使用 C 库的 malloc,还是会从系统堆里面申请内存!!

对于高端的单片机,有 MMU(内存管理单元) 模块,比如 Cortex-A 系列,有了MMU就能跑linux,那么内存管理也是必备。

四、结语(用还是不用?)

本文算是详细分析了一下 malloc 函数在单片机上的使用效果,我们知道了 malloc 函数使用申请了内存空间在哪里,我们也知道了内存碎片是如何产生的。

回到我们最初的问题,单片机领域,用还是不用 malloc 函数?

看完文章这个问题估计都不需要我直接回答了:

从项目复杂程度来说:

如果跑裸机 做些小项目,如果没有自己的内存管理方式不是必要都不建议使用,同时为了节约内存,可以把heap设置成很小(留一点给可能调用的C库函数会用到)。

如果跑操作系统,操作系统有完善的内存管理,可以痛快的使用操作系统的malloc接口函数。
但是如果做些小项目,也是可以不用的。

如果跑裸机 做些大项目???? 我个人不太建议……

从使用的芯片RAM大小来说:

如果你选用的芯片内存比较小,10多K甚至几K,还是用静态内存局部变量把,因为能够使用小内存的项目也不会太复杂,比如物联网传感器单品项目。

如果选用的芯片内存比较大上了 MB, 那么还是可以尝试使用 malloc 动态内存分配的,但是前提还是得有内存管理。

但是最后还是得说一下,随着现在的单片机发展,内存越来越大,虽然单片机小项目不建议使用 malloc 函数,但是我们上了操作系统以后,要学会去使用动态内存分配,因为当以后做的项目越来越复杂,线程越来越多,我们定义的局部变量越来越多,即便我们可以继续增大系统 栈 的大小,但是这终究不是一种合理的解决方式。
我们应该要学会合理的使用动态内存申请,为了以后向更高的地方前进 ~

没有必要钻牛角尖,如果项目简单但就是想用。比如我就一个函数使用malloc 申请动态内存,没有内存管理,我就是用了怎么地? 这种情况用不用都一样,看自己高兴,没有必要纠结!

谢谢!

本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

浅谈 malloc 函数在单片机上的应用 的相关文章

  • 除了 malloc/free 之外,程序还需要操作系统提供其他东西吗?

    我正在为我正在开发的操作系统设计内核 我实际上将其称为 核心 只是为了有所不同 但它基本上是相同的 如果我无法启动和运行多任务处理 内存管理和其他基本功能 那么操作系统本身的细节就无关紧要了 所以我需要首先解决这个问题 我有一些关于设计 m
  • 从c中的数组中删除偶数

    你好 我正在尝试大约 2 个小时来创建一个程序 该程序将从 c 中的动态分配数组 使用 malloc 中删除偶数 有人可以帮我提供一些提示或创建代码吗 附注这是我在这里的第一个主题 所以请随时给我一些关于如何正确发布问题的提示 假设您已经动
  • 程序不会因堆溢出而崩溃

    我写了以下程序 include
  • 为什么在C中使用malloc时要指定大小?

    采取以下代码 int p malloc 2 sizeof p p 0 10 Using the two spaces I p 1 20 allocated with malloc before p 2 30 Using another sp
  • alloca 函数崩溃,但 malloc 正常

    请解释一下崩溃的原因 我的第 3 方代码意外崩溃并出现 EXC BAD ACCESS static int overwrite selector struct srm target srm MARFileRemoverProgressBlo
  • 还有其他半/便携式方法来动态分配内存吗?

    this char buf NULL scanf ms buf 将获得动态分配的字符缓冲区 我知道这仅限于使用 Gcc 特别是 glibc 2 7 版本 编译的代码 我也知道 正确 和便携的方法是只使用malloc 和朋友 获取记忆 I m
  • C++ 中可以混合使用 free 和构造函数吗? [复制]

    这个问题在这里已经有答案了 可能的重复 调用free 或delete而不是delete 有什么危险吗 https stackoverflow com questions 1612031 is there any danger in call
  • sbrk 在 malloc.c 中如何/在何处使用?

    我在 高级 Unix 编程 以及其他几本书 中读到 Linuxmalloc 使用Linux系统调用sbrk 向操作系统请求内存 我正在看 glibcmalloc c代码 我可以看到很多提及sbrk 在注释中 但没有在代码中直接引用 如何 在
  • 为什么 C++ 需要对 malloc() 进行强制转换,而 C 不需要?

    我一直对此感到好奇 为什么在 C 中我必须转换返回值malloc但不是C语言 下面是 C 中有效的示例 int int ptr int malloc sizeof int 下面是 C 中的示例 该示例不起作用 无强制转换 int int p
  • Linux中分配特定地址

    我想在Linux进程中的特定地址分配一块内存 实际上我想做一些类似的事情 我会有进程号 每个进程都会调用库 由我编写 中的初始化函数 该函数将在进程的地址空间中分配一些内存 它将存储进程相关信息 这将由每个进程完成 一旦分配了该内存 程序就
  • glibc 已弃用的 __malloc_hook 功能的替代方案

    我正在为 C 编写一个内存分析器 并为此拦截对malloc realloc and free通过 malloc hooks 函数 不幸的是 这些已被弃用 因为它们在多线程环境中表现不佳 我找不到描述实现相同目标的替代最佳实践解决方案的文档
  • 为什么分配大块内存会失败,而重新分配小块内存却不会失败

    这段代码的结果是x指向一块大小为 100GB 的内存 include
  • 具有不同大小结构的结构数组的 malloc()

    如果每个结构都包含一个大小不同的字符串数组 那么如何正确地 malloc 一个结构数组 因此每个结构可能有不同的大小 并且不可能 realloc 结构体数量 sizeof 结构体名称 after malloc 初始大小 sizeof 结构名
  • mprotect 之后 malloc 导致分段错误

    在使用 mprotect 保护内存区域后第一次调用 malloc 时 我遇到分段错误 这是执行内存分配和保护的代码片段 define PAGESIZE 4096 void paalloc int size Allocates and ali
  • 如何解决malloc_error_break?

    我在 iOS 3 0 模拟器上遇到此错误 但在 3 1 3 和 3 2 模拟器上没有遇到此错误 创建符号断点后malloc error break 我在日志中看到了这一点 Session started at 2010 02 13 19 1
  • Visual C++ free 和 malloc 的线程安全性?

    有谁知道 free 和 malloc 在 Visual C 2010 上是否是线程安全的 我遇到了奇怪的问题 内存被损坏 我几乎认为这是唯一的可能性 有谁知道安全装置是否可以打开和关闭以及如何打开和关闭 前提是您链接的是线程安全库 http
  • 为什么这种故意错误地使用 strcpy 不会严重失败?

    为什么下面的C代码使用strcpy对我来说工作得很好吗 我试图通过两种方式让它失败 1 我尝试过strcpy从字符串文字到分配的内存 该内存太小而无法容纳它 它复制了整个事情并且没有抱怨 2 我尝试过strcpy来自一个不是的数组NUL 终
  • 嵌入式系统中的malloc [重复]

    这个问题在这里已经有答案了 我正在使用嵌入式系统 该应用程序在 AT91SAMxxxx 和 cortex m3 lpc17xxx 上运行 我正在研究动态内存分配 因为它会极大地改变应用程序的外观 并给我更多的力量 我认为我唯一真正的路线是为
  • malloc :匿名映射和魔法区域

    我只是在摆弄内存映射 想查看用户空间虚拟内存区域映射 写了一些像这样的行 char ptr NULL printf Allocating 300KB n ptr malloc 300 1024 printf Allocated at p s
  • 使用 cudamalloc()。为什么是双指针?

    我目前正在浏览有关的教程示例http code google com p stanford cs193g sp2010 http code google com p stanford cs193g sp2010 学习CUDA 演示的代码 g

随机推荐

  • nginx1.15安装和配置

    安装单机版nginx 首先下载安装包 nginx 1 15 0 tar gz pcre 8 38 tar gz 首先安装pcre tar zxvf pcre 8 38 tar gz cd pcre 8 38 configure prefix
  • STM32系列(HAL库)——单通道ADC转换

    本篇演示使用cubeMX配置STM32单通道ADC 话不多说 开始正文 一 前期准备 1 硬件 STM32C8T6最小系统板 USB TTL串口模块 ST Link下载器 2 软件 keil5 IDE cubeMX 二 cubeMX配置 1
  • 139. 单词拆分(DP)

    139 单词拆分 给你一个字符串 s 和一个字符串列表 wordDict 作为字典 请你判断是否可以利用字典中出现的单词拼接出 s 注意 不要求字典中出现的单词全部都使用 并且字典中的单词可以重复使用 示例 1 输入 s leetcode
  • 折半查找(C语言)

    折半查找 设定查找范围的下限low 上限high 由此确定查找范围的中间位置mid 中间位置的值等于待查的值 查找成功 中间位置的值小于待查的值 low mid 1 中间位置的值大于待查的值 high mid 1 直到low gt high
  • java基本语法 上

    目录 关键字与保留字 关键字 keyword 的定义和特点 保留字 标识符 Java中的名称命名规范 变量 变量的定义 变量的分类 整数类型 byte short int long 浮点类型 float double 字符类型 char 布
  • 函数与对象关系

    函数是对象 通过下面代码就能验证 var fn function console log fn instanceof Object true 函数跟对象之间的关系比较复杂 首先 所有对象都是通过函数创建的 有些人可能反驳 不对因为 var
  • AntD 可编辑行表格

    本地数据代码模板自用 官网例子改改 编辑行的自定义表格 import React useState from react import Table Input InputNumber Popconfirm Form Typography D
  • Umijs组件的初次应用

    在Umijs组件中我们不像vue那样可以看到组件的路径 在umijs刚刚搭建完成我们看不到组件的路径 所以我们要手动进行一些小小的改动 就可以看到了 一 应用前的准备 1 在当前项目文件路径下的终端输入 npm umi g page log
  • C 求整数n的二进制有多少个1

    首先 整数 分为正整数和负数 那么解题的时候就要注意这个正 负 提供了3中解题方案 第一种 int getBinary1 unsigned int n int count 0 while n if n 2 1 count n 2 retur
  • 说了那么多,到底什么是「神经搜索」?

    从人脑神经网络到深度学习神经网络 来自 Jina AI 的 Jack 将带大家从 0 到 1 认识 搜索 及 神经搜索 什么是神经搜索 它与普通搜索有什么区别 它能解决哪些问题 又有哪些优势和劣势 观看本期 Jina AI 小科普 在 Ja
  • SQL Server 问题集

    问题一 执行drop database xxx 反馈结果 无法对 数据库 xxx 执行 删除 因为它正用于复制 解决方案 只需要执行 sp removedbreplication XXX 然后再执行drop database xxx 原因分
  • vs2022提示函数不安全等问题

    我们在使用VS编译器编写C C 代码源程序的时候 发现在里面用scanf函数会编译出错 如下图所示 这个提示我们有两种方案可以解决 一是可以把scanf函数替换为 scanf s 如下 此时编译就会通过 但是在其他编译器里没有这个函数 因此
  • 【Figma技巧】Figma中快速制作斜线阴影的三种方法

    想要实现的效果 方法一 安装Hero Patterns插件 安装地址 https www figma com community plugin 743134103711120154 Hero Patterns for Figma 缺点 生成
  • openldap介绍和使用

    openldap介绍和使用 为什么会有本文 早期 公司是没有统一认证这个东西的 所以各自玩各自的 于是 confluence一个用户体系 gitlab一个用户体系 Jenkins一个用户体系等等 开发中要用到的开源软件数不胜数 每个软件都要
  • 齐纳二极管

    齐纳二极管 zener diodes 的主要作用就是当作一种电压调整器 QLCO A146提供稳定的参考电压 可应用在电源供应器 电压表与其他的仪器中 在本节中 你将会学习到齐纳二极管在适当的工作条件下 如何维持一个接近定值的直流电压 你将
  • HCL网络实操之ftp登录

    路由器配置ip in g0 0 ip address 192 168 56 200 24 路由器开启ftp服务 ftp server enable 创建用户 设置用户密码 设置权限 创建用户 local user cmoc 设置用户密码 p
  • JCreator 配置与用法

    对于一个web项目 里面由几个比较来说相对的标准目录 CSS 用来存放web项目中所使用到的样式文件 images 用来存放web项目中所使用的装饰图片 WEB INF 用来表示是一个jsp类型的项目 并且符合J2EE标准 inc 或者 i
  • 2023通信工程专业毕设题目大全

    文章目录 1前言 2 如何选题 3 通信工程选题方向 3 1 移动通信方向 3 2 嵌入式开发方向 3 3 人工智能方向 3 4 物联网方向 3 5 算法研究方向 3 6 移动应用开发方向 3 7 网络通信方向 3 8 学长作品展示 4 最
  • redis后端启动及关闭

    本文笔记为传智教程视频截图整理 redis前端启动 无法部署集群
  • 浅谈 malloc 函数在单片机上的应用

    聊聊 malloc函数 在单片机程序设计中怎么使用 目录 前言 一 malloc 函数简介 二 malloc 之于单片机 2 1 malloc 函数申请的内存在哪里 2 2 用与不用malloc的区别 三 malloc可能遇到的问题 3 1