stm32的const数据位于Flash上,Flash会比RAM大很多倍

2022-05-10     大方老师单片机

原标题:stm32的const数据位于Flash上,Flash会比RAM大很多倍

stm32const数据位FlashFlashRAM大很多

///插播一条:我自己在今年年初录制了一套还比较系统的入门单片机教程,想要的同学找我拿就行了免费的,私信我就可以~点我头像黑色字体加我地球呺也能领取哦。最近比较闲,带做毕设,带学生参加省级或以上比///

01首先聊一聊

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

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

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

那么主控芯片换不了我们还有什么办法呢?那我们应该从原本的程序中挤出资源来使用了,下面我总结了几种常总方法供大家参照。

02共联-union

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

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

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

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

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

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

#include 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来存储就简直浪费7bit,假如以后也有类似的的情况,那么大局部内存都得不到有效的应用。所C语言的位域就是用来攻克这个问题。

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

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

2)没名位域的使用,能够看下面的代码。

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

#include//结果:编译通//理由:常规形式(构造体占用两个字节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; }

04构造对齐

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

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

#include#pragma pack(1)

//有字节对齐预编译结果为128

//没字节对齐预编译结果为66

typedef 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"); }

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

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

06const

1const的使用

const的用法应该是老生常谈的知识点了,假如还有不是特别清楚的小搭档能够参<一文搞Cconst关键>,bug菌就不反复造轮子了,直接stm32单片机为例看const变量的的存储方式。

运行结果分析一下:

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

?程序编译成功以后,就能够直接map文件中查const修饰的数组名。

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

2const数据的存储

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

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

上图来源STMemory Mapping

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

3const数据使用

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

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

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

所以,一句话说得"前人栽树,后人乘凉;前人挖坑,后人fen"。前面我们分析stm32const数据位Flash上,一Flash都会RAM大上好几倍。

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

注意一点的 :访RAM一般来说会比访Flash要快一些,然而大局部项目对于

文章来源: https://twgreatdaily.com/zh-hans/e13f893a1b5be09c374f3140a252922e.html