0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

必看干货:单片机节省内存的方法

工程师 来源:最后一个bug 作者:最后一个bug 2020-09-11 17:13 次阅读

来源:最后一个bug

作者:bug菌

首先聊一聊

大家都知道进行单片机编程和计算机编程有个最大的差别就是单片机的资源非常的有限,并且对于大部分低端单片机而言都没有操作系统。除了一些嵌入式级的芯片用了Linux系统外,其他大部分操作都是比较简单的RTOS,可能还有一些简单的应用或者芯片根本不用系统,直接是裸机程序。

不过大部分单片机编程都与硬件密切的结合,这样工程师能够对当前的项目对象有更多的把控能力和理解能力。但是由于它的简单,我们平时在工作中往往需要控制一个项目的成本,对于单片机的选型和资源的评估都是非常谨慎;同样随着我们项目功能的不断扩展,也会让系统程序逐步变得庞大,这时候资源的使用就更需要节约点用了。

所以当资源受限制(一般的单片机RAM也就Kb级别),比如说单片机RAM不够了,即使你有再牛的算法可能也无法加入到项目中来,那么有些同志们会问,那换芯片不就可以了吗?我只想说这位同志你想多了,对于不怎么热卖产品或者不规范的公司可能还允许你试一试,可是一般的公司项目卡着走的,换了主控芯片,暂且不说软件上的移植工作,换了芯片成本上必定增加,产品的测试都得重新规划,老板领导可不愿意了。

那么主控芯片换不了我们还有什么办法呢?那我们应该从原本的程序中挤出资源来使用了,下面我总结了几种常总方法供大家参考。(具体内容可以网络查找)

02

共联体-union

union-共联体,是C语言常用得关键字。从字面上的意思就是共同联合在一起的意思,union所有的成员共同维护一段能够内存空间,其内存的大小取决于所有成员中占用空间最大的成员。

union结构体由于是共用同一片内存可以大大节省内存空间,那一般什么情况下使用union?又或者union还有什么特点?下面我将用几点为大家解答。

1)所有的union的成员及本身的地址是一样的。

2)union的存储模型受大小端的影响,我们可以通过下面的代码进行测试。(如果输出结果为1,表示小端模式,否则为大端模式)

大端模式(Big_endian):一个数据的高字节存储在低地址,低字节存储在高地址。其指针指向的首地址位于低地址。

小端模式(Little_endian):一个数据的高字节存储在高地址,低字节存储在低地址。其指针指向的首地址位于高地址。

3)union不同于结构体struct,union对成员的改变可能会影响到其他成员变量,所以我们要形成一种互斥使用,比如说我们的顺序执行其实就是每个代码都是互斥的,所以我们可以用union进行函数处理缓存等。(个人觉得也可以认为是分时复用,并且是不会受内存初值影响的处理)

#include《stdio.h》typedef union _tag_test{ char a; int b;}uTest;uTest test;unsigned char Checktype(void);int main(void){ printf(“%x\n”,(unsigned int)&test.a); printf(“%x\n”,(unsigned int)&test.b); printf(“%x\n”,(unsigned int)&test); printf(“%d\n”,Checktype()); } unsigned char Checktype(void){ uTest chk; chk.b = 0x01; if(chk.a == 0x01)return 1; return 0; }

03

位域

位域可能对于初学者用得比较少,不过对于大部分参加工作的工程师应该屡见不鲜了,确实它也是我们省内存的神器。

因为在我们平时编程过程中,我们使用的变量与实际情况是息息相关的,就比如说开关的状态,我们一般就是0或者是1分别表示打开和关闭,那么我们用一个bit就能表示,假如说我们用一个char来存储就几乎浪费了7个bit,如果以后也有类似的的情况,那么大部分内存都得不到有效的应用。所以C语言的位域就是用来解决这个问题。

不过我们需要注意如下几点:

1)位域是在结构体中实现的,其中位域规定的长度不能超过所定义类型,且一个位域只能定义在同一个存储单元中。

2)无名位域的使用,可以看下面的代码。

3)由于位域与数据类型有关系,那么他的内存占用情况也与平台的位数相关。(相关内容可网络查找)

#include《stdio.h》//结果:编译通过//原因:常规形式(结构体占用两个字节)typedef struct _tag_test1{ char a:1; char b:1; char c:1; char d:6;}sTest1;//结果:编译无法通过//原因:d的位域长度10超过了char类型长度/*typedef struct _tag_test2{ char a:1; char b:1; char c:1; char d:10;}sTest2;*///结果:编译可通过//原因:下面使用无名位域,且占8个字节typedef struct _tag_test3{ int a:1; int b:1; int :0;//无名位域 int c:1;}sTest3;int main(void){ printf(“%d\n”,sizeof(sTest1)); printf(“%d\n”,sizeof(sTest3)); printf(“欢迎关注公众号:最后一个bug\n”); }

04结构体对齐

结构体对齐问题可能大部分人关注的不是很多,可能在通讯领域进行内存的copy时候接触得比较多。结构体对齐问题也是与平台相关,CPU为了提高访问内存的效率,一次性可能读取2个字节,4个字节,8个字节等,所以编译器会自动对结构体内存进行对齐。

废话不多说,代码说明一切:

#include《stdio.h》#pragma pack(1)//有字节对齐预编译结果为:12,8//无字节对齐预编译结果为:6,6typedef struct _tag_test1{ char a; int b; char c;}STest1;typedef struct _tag_test2{ int b; char a; char c;}STest2;int main(void){ printf(“%d\n”,sizeof(STest1)); printf(“%d\n”,sizeof(STest2)); printf(“欢迎关注公众号:最后一个bug\n”); }

05

算法优化

算法优化其实主要是我们通过修改一些算法的实现一种效率与内存使用的一个平衡,我们都知道我们的算法都存在着复杂度的问题,我们大部分高效率的算法都是通过使用内存来换效率,也就是一种用空间换时间的概念。那么当我们内存使用有限的时候我们可以适当的用时间来换空间的方法,腾出更多的空间来实现更多的功能。

同样我们在进行相关设计的时候可以尽量使用局部变量来减少全局变量的使用!

06

利用const

1、const的使用

关于const的用法应该是老生常谈的知识点了,如果还有不是特别清楚的小伙伴可以参考《 一文搞定C语言const关键字 》一文,bug菌就不重复造轮子了,直接以stm32单片机为例看看const变量的的存储方式。

参考demo:

#include “led.h” #include “delay.h” #include “usart.h” #define DEV_NUM_MAX (3) #define DEV_PARAM_MAX (2) typedef struct _tag_DevParam{ char* Name; //设备名称 uint32_t Param[DEV_PARAM_MAX]; //设备参数}sDevParam; const sDevParam stDevParam[DEV_NUM_MAX] = { {“Uart1”,57600,0}, {“Uart2”,57600,1}, {“CAN”,1000000,0}, };/*************************************** * Fuction:const内存分配测试 * Author :bug菌 **************************************/ int main(void) { uint8_t t = 0; uint8_t devCnt = 0; delay_init(); //延时函数初始化 uart_init(115200); //串口初始化 printf(“\n*******************const Test*******************\r\n”); for(devCnt = 0 ;devCnt 《 DEV_NUM_MAX;devCnt++) { printf(“DevName = %s,Param1 = %d,Param2 = %d\r\n”,stDevParam[devCnt].Name,\ stDevParam[devCnt].Param[0],\ stDevParam[devCnt].Param[1]); } printf(“stDevParam Size : %d \r\n”,sizeof(stDevParam)); printf(“stDevParam Addr : 0x%X \r\n”,stDevParam); printf(“\n***********欢迎关注公众号:最后一个bug************\n”); while(1) { delay_ms(10); if(++t 》 150){LED0=0;}else{LED0=1;} } }

运行结果:

分析一下:

对于stm32的所有存储映像都在对应工程所编译生成的.map文件中,对.map文件(其文件在工程目录中)的熟悉度就在一定程度上彰显你对stm32单片机的熟练程度。

程序编译成功以后,就可以直接在map文件中查找const修饰的数组名,从而得到如下结果:

从上图我们了解到其stDevParam变量位置0x080016b8数据区且位于(.contdata段--只读数据段)并占用了36个字节,与我们串口输出结果是相符合的。

2、const数据的存储

通过上面的测试程序显示了const数据的存储位置,那么我们看一下该位置位于stm32的哪块存储区域,是RAM还是FLASH?

因为我们节省内存主要就是通过占用更小的RAM来实现相同的项目需求,那么对于MCU而言最好就是的借助Flash,通过时间来置换空间,拿出对应的数据手册看看这些存储范围是如何分配的。

上图来源于ST手册Memory Mapping

很明显前一节测试的const stDevParam变量位置0x080016b8处,正好处于FLASH存储位置,所以其并没有占用RAM资源。

3、const数据使用

很多写单片机程序的小伙伴都喜欢把一些只读的变量用全局变量来保存,然而这些变量基本上只保存一些参数,这对于单片机的RAM资源是非常浪费的。

bug菌曾经接手过一个前同事项目,怎么说呢?可能这个项目他也是接手别人的,该项目MCU还外部扩展了一个16M的SDRAM,大家都觉得反正RAM大,变量随便定,也不去管数据范围,动不动就float,double,真的是牛。

直到bug菌接手内存占用率已高达95%,后面稍微添加一些需求感觉RAM就要爆掉了,没办法这样下去终究会出问题,于是申请了代码重构,通过优化代码结构、设计等RAM占用率直接降到了50%左右,可以想象一下之前的开发人员是多么的任性。

所以一句话说得好“前人栽树,后人乘凉;前人挖坑,后人入fen”。前面我们分析了stm32的const数据位于Flash上,一般Flash都会比RAM打上好几倍:(如下图所示:)

上图来源于ST官网

这样对于一些预先设置好的参数等等都可以整理以后统一放到类似于前面demo中这样的结构体数组中,从而可以大大减少对RAM的占用。

注意一点的是 : 访问RAM一般来说会比访问Flash要快一些,然而大部分项目对于这样的差异影响非常之小,后面bug菌会为大家再带来一篇文章讲讲这块的知识。

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 单片机
    +关注

    关注

    6036

    文章

    44557

    浏览量

    635103
  • 编程
    +关注

    关注

    88

    文章

    3615

    浏览量

    93716
  • CONST
    +关注

    关注

    0

    文章

    44

    浏览量

    8163
收藏 人收藏

    评论

    相关推荐

    单片机Debug工具性能对比 单片机调试常用命令

    。以下是对单片机调试工具性能的简要对比以及一些常用的调试命令。 单片机Debug工具性能对比 Keil uVision 性能 :Keil uVision 是一款功能强大的集成开发环境(IDE),支持多种单片机,特别是ARM Co
    的头像 发表于 12-19 09:56 199次阅读

    单片机调试常见问题与解决方法

    烧录到单片机时,烧录软件显示无法与单片机通信。 解决方法: 检查单片机的电源是否正常,确保电源电压在规定的范围内。 检查烧录线的连接是否正确,包括数据线和地线。 确认烧录软件是否选择了
    的头像 发表于 11-01 14:11 742次阅读

    单片机的中断机制

    单片机的中断机制是一种重要的处理方式,它允许单片机在执行主程序的过程中,能够暂停当前任务,转而处理外部或内部紧急事件。这种机制极大地提高了系统的响应速度和处理能力,使得单片机在各种应用领域中得到广泛应用。以下是对
    的头像 发表于 10-17 18:03 671次阅读

    单片机烧录程序的线比单片机上的少还能烧录吗

    的存储器通常分为两类:ROM(只读存储器)和RAM(随机存取存储器)。ROM用于存储程序代码,而RAM用于存储程序运行过程中的数据。烧录过程就是将程序代码写入ROM中。 单片机烧录方法 单片机烧录的
    的头像 发表于 09-02 09:54 498次阅读

    单片机boot0和boot1怎么设置

    单片机的启动模式通常包括从内部ROM启动、从外部ROM启动、从外部Flash启动等。 不同的启动模式对应不同的Boot0和Boot1设置。 Boot0和Boot1的设置方法 Boot0和Boot1通常通过
    的头像 发表于 08-22 09:50 2501次阅读

    单片机中的几种环形缓冲区的分析和实现

    单片机中的几种环形缓冲区的分析和实现一、简介环形缓冲区(RingBuffer)是一种高效的使用内存方法,它将一段固定长度的内存看成一个环形结构,用于存储数据,能够避免使用动态申请
    的头像 发表于 08-14 08:39 848次阅读
    <b class='flag-5'>单片机</b>中的几种环形缓冲区的分析和实现

    单片机内存和程序大小有什么关系吗?怎么选用不同内存大小的单片机

    单片机内存和程序大小有什么关系吗?怎么选用不同内存大小的单片机?仅从成本考虑吗?
    发表于 05-16 06:03

    你知道吗? 51单片机也有动态内存分配

    、realloc、free。他们的头文件在中,所以使用内存管理必须包含该头文件。二、使用方法51单片机需要使用内存管理API必须要手动调用初始化堆
    的头像 发表于 04-26 08:10 1537次阅读
    你知道吗? 51<b class='flag-5'>单片机</b>也有动态<b class='flag-5'>内存</b>分配

    51单片机串口通信详细介绍-小白必看

    51单片机串口通信是单片机与外部设备通信的重要方式,它基于串行通信的原理,以字节为单位进行数据传输。在51单片机中,串口通信主要依赖于两个关键寄存器:SBUF寄存器和SCON寄存器。SBUF寄存器用于存储要发送或接收的数据,而S
    的头像 发表于 04-17 09:33 1987次阅读
    51<b class='flag-5'>单片机</b>串口通信详细介绍-小白<b class='flag-5'>必看</b>

    如何系统、科学地自学单片机

    的自学单片机呢?自学单片机需要一定的计划和方法,以下是具体的步骤和建议。如何系统、科学地自学单片机?学习电子基础知识:理解电路原理、数字电子技术、模拟电子技术等基础
    的头像 发表于 03-28 08:03 1081次阅读
    如何系统、科学地自学<b class='flag-5'>单片机</b>?

    单片机编程和plc编程有什么区别

    编程的基本概念 单片机是一种在一个芯片上集成了处理器核心、内存、输入输出接口等功能的微控制器。单片机通常用于控制与外部设备的交互,如传感器、电机、键盘等。单片机编程是将程序代码写入芯片
    的头像 发表于 02-22 10:23 2729次阅读

    单片机的最小系统由什么组成 单片机的最小系统包括哪些

    单片机(Microcontroller Unit,MCU)的最小系统包括以下几个主要组成部分: 单片机芯片:单片机是整个最小系统的核心部分,它集成了中央处理器(Central Processing
    的头像 发表于 02-02 11:27 1.1w次阅读

    基于单片机控制的交通灯设计

    本课程设计是在学完单片机原理及课程之后综合利用所学单片机知识完成一个单片机应用系统设计并在实验室实现。
    的头像 发表于 01-22 15:31 2156次阅读
    基于<b class='flag-5'>单片机</b>控制的交通灯设计

    单片机通过USB升级固件的方法

    单片机升级固件的方法有很多中,比如:ISP(在系统编程)、ICP(在电路编程)、IAP(在应用编程)等。
    的头像 发表于 01-11 09:37 1973次阅读
    <b class='flag-5'>单片机</b>通过USB升级固件的<b class='flag-5'>方法</b>

    stc51单片机怎么烧录

    步骤 常见问题与解决方法 STC51单片机是一种常见的8位单片机,广泛应用于各种嵌入式系统。它具有高性能、低成本、易于学习和使用等优点。STC51单片机使用汇编语言或C语言进行编程,可
    的头像 发表于 01-02 17:41 3414次阅读