當前位置:首頁 > 公眾號精選 > 嵌入式微處理器
[導讀]01 首先聊一聊 大家都知道進行單片機編程和計算機編程有個最大的差別就是單片機的資源非常的有限,并且對于大部分低端單片機而言都沒有操作系統(tǒng)。除了一些嵌入式級的芯片用了Linux系統(tǒng)外,其他大部分操作都是比較簡單的RTOS,可能還有一些簡單的應用或者芯片

01 首先聊一聊


大家都知道進行單片機編程和計算機編程有個最大的差別就是單片機的資源非常的有限,并且對于大部分低端單片機而言都沒有操作系統(tǒng)。除了一些嵌入式級的芯片用了Linux系統(tǒng)外,其他大部分操作都是比較簡單的RTOS,可能還有一些簡單的應用或者芯片根本不用系統(tǒng),直接是裸機程序。
不過,大部分單片機編程都與硬件密切的結(jié)合,這樣工程師能夠?qū)Ξ斍暗捻椖繉ο笥懈嗟陌芽啬芰屠斫饽芰Α5怯捎谒暮唵?,我們平時在工作中往往需要控制一個項目的成本,對于單片機的選型和資源的評估都是非常謹慎;同樣隨著我們項目功能的不斷擴展,也會讓系統(tǒng)程序逐步變得龐大,這時候資源的使用就更需要節(jié)約點用了。
所以當資源受限制(一般的單片機RAM也就Kb級別),比如說單片機RAM不夠了,即使你有再牛的算法可能也無法加入到項目中來,那么有些同志們會問,那換芯片不就可以了嗎?我只想說這位同志你想多了,對于不怎么熱賣產(chǎn)品或者不規(guī)范的公司可能還允許你試一試,可是一般的公司項目卡著走的,換了主控芯片,暫且不說軟件上的移植工作,換了芯片成本上必定增加,產(chǎn)品的測試都得重新規(guī)劃,老板領(lǐng)導可不愿意了。
那么主控芯片換不了我們還有什么辦法呢?那我們應該從原本的程序中擠出資源來使用了,下面我總結(jié)了幾種常總方法供大家參考。(具體內(nèi)容可以網(wǎng)絡(luò)查找)

02 共聯(lián)體-union


union-共聯(lián)體,是C語言常用得關(guān)鍵字。從字面上的意思就是共同聯(lián)合在一起的意思,union所有的成員共同維護一段能夠內(nèi)存空間,其內(nèi)存的大小取決于所有成員中占用空間最大的成員。
union結(jié)構(gòu)體由于是共用同一片內(nèi)存可以大大節(jié)省內(nèi)存空間,那一般什么情況下使用union?又或者union還有什么特點?下面我將用幾點為大家解答。
1)所有的union的成員及本身的地址是一樣的。
2)union的存儲模型受大小端的影響,我們可以通過下面的代碼進行測試。(如果輸出結(jié)果為1,表示小端模式,否則為大端模式)
大端模式(Big_endian):一個數(shù)據(jù)的高字節(jié)存儲在低地址,低字節(jié)存儲在高地址。其指針指向的首地址位于低地址。 小端模式(Little_endian):一個數(shù)據(jù)的高字節(jié)存儲在高地址,低字節(jié)存儲在低地址。其指針指向的首地址位于高地址。
3)union不同于結(jié)構(gòu)體struct,union對成員的改變可能會影響到其他成員變量,所以我們要形成一種互斥使用,比如說我們的順序執(zhí)行其實就是每個代碼都是互斥的,所以我們可以用union進行函數(shù)處理緩存等。(個人覺得也可以認為是分時復用,并且是不會受內(nèi)存初值影響的處理)
#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 位域


位域可能對于初學者用得比較少,不過對于大部分參加工作的工程師應該屢見不鮮了,確實它也是我們省內(nèi)存的神器。
因為在我們平時編程過程中,我們使用的變量與實際情況是息息相關(guān)的,就比如說開關(guān)的狀態(tài),我們一般就是0或者是1分別表示打開和關(guān)閉,那么我們用一個bit就能表示,假如說我們用一個char來存儲就幾乎浪費了7個bit,如果以后也有類似的的情況,那么大部分內(nèi)存都得不到有效的應用。所以C語言的位域就是用來解決這個問題。
不過,我們需要注意如下幾點:
1)位域是在結(jié)構(gòu)體中實現(xiàn)的,其中位域規(guī)定的長度不能超過所定義類型,且一個位域只能定義在同一個存儲單元中。
2)無名位域的使用,可以看下面的代碼。
3)由于位域與數(shù)據(jù)類型有關(guān)系,那么他的內(nèi)存占用情況也與平臺的位數(shù)相關(guān)。(相關(guān)內(nèi)容可網(wǎng)絡(luò)查找)
#include//結(jié)果:編譯通過//原因:常規(guī)形式(結(jié)構(gòu)體占用兩個字節(jié))typedef struct _tag_test1{ char a:1; char b:1; char c:1; char d:6;}sTest1;//結(jié)果:編譯無法通過//原因:d的位域長度10超過了char類型長度/*typedef struct _tag_test2{ char a:1; char b:1; char c:1; char d:10;}sTest2;*///結(jié)果:編譯可通過//原因:下面使用無名位域,且占8個字節(jié)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("歡迎關(guān)注公眾號:最后一個bug\n"); }

		
		

04 結(jié)構(gòu)體對齊


結(jié)構(gòu)體對齊問題可能大部分人關(guān)注的不是很多,可能在通訊領(lǐng)域進行內(nèi)存的copy時候接觸得比較多。結(jié)構(gòu)體對齊問題也是與平臺相關(guān),CPU為了提高訪問內(nèi)存的效率,一次性可能讀取2個字節(jié),4個字節(jié),8個字節(jié)等,所以編譯器會自動對結(jié)構(gòu)體內(nèi)存進行對齊。
廢話不多說,代碼說明一切:
#include#pragma pack(1)//有字節(jié)對齊預編譯結(jié)果為:12,8//無字節(jié)對齊預編譯結(jié)果為: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("歡迎關(guān)注公眾號:最后一個bug\n"); }

05 算法優(yōu)化


算法優(yōu)化其實主要是我們通過修改一些算法的實現(xiàn)一種效率與內(nèi)存使用的一個平衡,我們都知道我們的算法都存在著復雜度的問題,我們大部分高效率的算法都是通過使用內(nèi)存來換效率,也就是一種用空間換時間的概念。那么當我們內(nèi)存使用有限的時候我們可以適當?shù)挠脮r間來換空間的方法,騰出更多的空間來實現(xiàn)更多的功能。
同樣,我們在進行相關(guān)設(shè)計的時候可以盡量使用局部變量來減少全局變量的使用!

06 利用const

1、const的使用
關(guān)于const的用法應該是老生常談的知識點了,如果還有不是特別清楚的小伙伴可以參考< 一文搞定C語言const關(guān)鍵字 >一文,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; //設(shè)備名稱 uint32_t Param[DEV_PARAM_MAX]; //設(shè)備參數(shù)}sDevParam;   const sDevParam stDevParam[DEV_NUM_MAX] = { {"Uart1",57600,0}, {"Uart2",57600,1}, {"CAN",1000000,0}, };/*************************************** * Fuction:const內(nèi)存分配測試 * Author :bug菌  **************************************/ int main(void) { uint8_t t = 0; uint8_t devCnt = 0;  delay_init(); //延時函數(shù)初始化  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***********歡迎關(guān)注公眾號:最后一個bug************\n"); while(1) { delay_ms(10);  if(++t > 150){LED0=0;}else{LED0=1;} }  }

運行結(jié)果分析一下:
  • 對于stm32的所有存儲映像都在對應工程所編譯生成的.map文件中,對.map文件(其文件在工程目錄中)的熟悉度就在一定程度上彰顯你對stm32單片機的熟練程度。
  • 程序編譯成功以后,就可以直接在map文件中查找const修飾的數(shù)組名。
  • 從上面我們了解到其stDevParam變量位置0x080016b8數(shù)據(jù)區(qū)且位于(.contdata段--只讀數(shù)據(jù)段)并占用了36個字節(jié),與我們串口輸出結(jié)果是相符合的。

2、const數(shù)據(jù)的存儲
通過上面的測試程序顯示了const數(shù)據(jù)的存儲位置,那么我們看一下該位置位于stm32的哪塊存儲區(qū)域,是RAM還是FLASH?
因為我們節(jié)省內(nèi)存主要就是通過占用更小的RAM來實現(xiàn)相同的項目需求,那么對于MCU而言最好就是的借助Flash,通過時間來置換空間,拿出對應的數(shù)據(jù)手冊看看這些存儲范圍是如何分配的。
上圖來源于ST手冊Memory Mapping
很明顯前一節(jié)測試的const stDevParam變量位置0x080016b8處,正好處于FLASH存儲位置,所以其并沒有占用RAM資源。
3、const數(shù)據(jù)使用
很多寫單片機程序的小伙伴都喜歡把一些只讀的變量用全局變量來保存,然而這些變量基本上只保存一些參數(shù),這對于單片機的RAM資源是非常浪費的。
bug菌曾經(jīng)接手過一個前同事項目,怎么說呢?可能這個項目他也是接手別人的,該項目MCU還外部擴展了一個16M的SDRAM,大家都覺得反正RAM大,變量隨便定,也不去管數(shù)據(jù)范圍,動不動就float,double,真的是牛。
直到bug菌接手內(nèi)存占用率已高達95%,后面稍微添加一些需求感覺RAM就要爆掉了,沒辦法這樣下去終究會出問題,于是申請了代碼重構(gòu),通過優(yōu)化代碼結(jié)構(gòu)、設(shè)計等RAM占用率直接降到了50%左右,可以想象一下之前的開發(fā)人員是多么的任性。
所以,一句話說得好"前人栽樹,后人乘涼;前人挖坑,后人入fen"。前面我們分析了stm32的const數(shù)據(jù)位于Flash上,一般Flash都會比RAM打上好幾倍。
這樣對于一些預先設(shè)置好的參數(shù)等等都可以整理以后統(tǒng)一放到類似于前面demo中這樣的結(jié)構(gòu)體數(shù)組中,從而可以大大減少對RAM的占用。
注意一點的是 : 訪問RAM一般來說會比訪問Flash要快一些,然而大部分項目對于這樣的差異影響非常之小,后面bug菌會為大家再帶來一篇文章講講這塊的知識。

-END-


本文授權(quán)轉(zhuǎn)載自最后一個bug,作者:bug菌




推薦閱讀



【01】怎么學習單片機外圍器件?

【02】漫畫版:如何學習單片機?

【03】單片機:3種時鐘電路方案對比,你常用哪一種?

【04】單片機編程技術(shù)學習攻略

【05】國產(chǎn)超低價單片機五宗罪!“扶不起”的原因就是它們?


免責聲明:整理文章為傳播相關(guān)技術(shù),版權(quán)歸原作者所有,如有侵權(quán),請聯(lián)系刪除

免責聲明:本文內(nèi)容由21ic獲得授權(quán)后發(fā)布,版權(quán)歸原作者所有,本平臺僅提供信息存儲服務(wù)。文章僅代表作者個人觀點,不代表本平臺立場,如有問題,請聯(lián)系我們,謝謝!

嵌入式ARM

掃描二維碼,關(guān)注更多精彩內(nèi)容

本站聲明: 本文章由作者或相關(guān)機構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點,本站亦不保證或承諾內(nèi)容真實性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫毥谦F公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動 BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運行,同時企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風險,如企業(yè)系統(tǒng)復雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機 衛(wèi)星通信

要點: 有效應對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅持高質(zhì)量發(fā)展策略,塑強核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運營商 數(shù)字經(jīng)濟

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動力信息技術(shù)(集團)股份有限公司(以下簡稱"軟通動力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉