在线观看www成人影院-在线观看www日本免费网站-在线观看www视频-在线观看操-欧美18在线-欧美1级

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

在STM32中為什么要引入鏈表?

GReq_mcu168 ? 來源:知曉編程 ? 作者:知曉編程 ? 2021-05-03 10:30 ? 次閱讀

1、為何引入鏈表

在程序中經常面臨一個問題,我們需要保存一定數量的對象,但是對象數目是不確定的,或者說是隨時增加或減少的。這時候最簡單的方法是創建一個足夠大的數組,用來存儲這些對象。我最近開發一個項目就遇到類似的問題,下面我把問題簡化一下。

需求:通過PC下發一些矩形的坐標和寬高信息,每個區域有個ID編號,并在這些矩形內填充一定的數據。

通常情況下,最簡單易懂的做法是,限制最多5個區域,每個區域存儲1K數據。因此設置了這樣的一個結構體(類似于面向對象語言里說的成員屬性)。

typedef struct Area_Inf{ uint8_t ID; uint8_t X; uint8_t Y; uint8_t Width; uint8_t Height; uint8_t data_len;}Area_Inf_Typedef;

然后定義結構體的實體。

#define Area_Num 5#define Area_cache 1024

Area_Inf_Typedef Area_Info[Area_Num];uint8_t Area_Data[Area_Num*Area_cache];//存儲區域的數據

/*找到ID為5的區域,并將數據拷貝出去*/void main(){ uint8_t i; uint8_t data[1024]; for(i = 0;i 《 Area_Num;i++) { if(Area_Info[i].ID == 5) { memcpy(data,&Area_Data[i*Area_cache ],Area_Info[i].data_len); } }}

上面這種做法是最簡單易懂的,但不靈活,比如有客戶要求10個區域,但是每個區域存儲的數據很少,根本用不到1K。雖然上面的程序已經使用了宏定義,只需要修改宏定義就能實現要求。但這意味著不同的客戶,需要編譯不同的固件。

#define Area_Num 10#define Area_cache 512

這樣的程序存在的問題:

1、在內存資源很緊缺的單片機程序中,當區域數據很少時,這種程序的處理方法浪費了大量的內存空間。

2、數值固定,需要存儲更多區域,即使還有內存,還是需要修改宏定義,重新編譯固件,不靈活。

這時需要引入鏈表來解決這個問題。

2、鏈表實現

鏈表實際上是線性表的鏈式存儲結構,與數組不同的是,它是用一組任意的存儲單元來存儲線性表中的數據,存儲單元不一定是連續的,且鏈表的長度不是固定的,鏈表數據的這一特點使其可以非常的方便地實現節點的插入和刪除操作。鏈表的每個元素稱為一個節點,每個節點都可以存儲在內存中的不同的位置,為了表示每個元素與后繼元素的邏輯關系,以便構成“一個節點鏈著一個節點”的鏈式存儲結構,除了存儲元素本身的信息外,還要存儲其直接后繼信息,因此,每個節點都包含兩個部分,第一部分稱為鏈表的數據區域,用于存儲元素本身的數據信息。

6ee7ead4-9e2c-11eb-8b86-12bb97331649.png

對于上面的問題,我們使用鏈表解決,需要配合內存管理才能實現。內存管理這一塊,大家可以自己編寫內存管理驅動,也可以使用C庫的malloc和free函數。如何字節編寫內存管理驅動不是本文的重點,下文將使用C庫的malloc和free函數進行內存管理。

使用鏈表的方式,在原有的成員屬性結構體的前提上,還要再封裝多一層鏈表管理。以單向鏈表為例:

typedef struct Area_Inf{ uint8_t ID; uint8_t X; uint8_t Y; uint8_t Width; uint8_t Height; uint8_t data_len; uint8_t* Area_Data;}Area_Inf_Typedef;

typedef struct Area_List_Inf{ Area_Inf_Typedef *Area_Inf; struct Area_List_Inf *next_Area_Inf; //用于指向下一個}Area_List_Inf_Typedef;

Area_List_Inf_Typedef *Head_Area_List; //鏈表的頭指針

由于在定義的時候,只定義了一個頭指針,那么它也只是個指向了Area_List_Inf_Typedef也就是鏈表結構體的指針,同樣沒有內存空間,在沒有創建新增鏈表之前,它是一個野指針。

所以,在具體應用之前,需要先執行一個初始化操作,也就是申請空間給鏈表管理結構體,然后頭指針指向這個空間。

/*** @brief 動態區鏈表初始化* @return int */int Area_List_Init(void){ //申請鏈表類型大小的空間,并讓頭指針指向它 Head_Area_List = (Area_List_Inf_Typedef*)malloc(sizeof(Area_List_Inf_Typedef)); if(Head_Area_List == NULL) return false; //同時要標記下一個信息為空 Head_Area_List-》next_Area_Inf = NULL; return true;}

通過PC下發一個新的區域信息后,增加新區域到鏈表末尾。

/*** @brief 在鏈表末尾增加一個區域參數* @param Area_Inf 增加的區域區參數指針* @return int */int Add_Area_ToList(Area_Inf_Typedef *Area_Inf){ Area_List_Inf_Typedef *p = Head_Area_List; while(p-》next_Area_Inf!=NULL) { p = p-》next_Area_Inf; } //先申請鏈表結構體的空間,因為后續還要繼續增加 p-》next_Area_Inf = (Area_List_Inf_Typedef*)malloc(sizeof(Area_List_Inf_Typedef)); if(p-》next_Area_Inf == NULL) return false;//申請不到內存,返回失敗 //指向剛剛申請的空間,并為需要存放的動態區信息申請對應的內存 p = p-》next_Area_Inf; p-》Area_Inf = (Area_Inf_Typedef*)malloc(sizeof(Area_Inf_Typedef)); if(p-》Area_Inf == NULL) { free(p);//由于申請失敗,先前申請的鏈表空間也要釋放 return false; } memcpy(p-》Area_Inf,Area_Inf,sizeof(Area_Inf_Typedef)); /*拷貝數據*/ p-》Area_Inf-》Area_Data = (uint8_t*)malloc(Area_Inf-》data_len); if(p-》Area_Inf-》Area_Data == NULL) { free(p-》Area_Inf); free(p); return false; } memcpy(p-》Area_Inf-》Area_Data,Area_Inf-》Area_Data,Area_Inf-》data_len); //標記這個鏈表的尾部 p-》next_Area_Inf=NULL; //添加成功 return true;}

通過PC下發一個刪除指定ID的區域命令。

/*** @brief 根據區域ID刪除動態區* @param num 區域ID* @return int */int Delete_Area_Accordingn_ID(int num){ int res = false; Area_List_Inf_Typedef *p = Head_Area_List; while(p-》next_Area_Inf!=NULL) { Area_List_Inf_Typedef *temp = p; p = p-》next_Area_Inf; if(p-》Area_Inf-》ID == num)//匹配到對應的值 { temp-》next_Area_Inf = p-》next_Area_Inf; //釋放內存空間 free(p-》Area_Inf-》Area_Data); free(p-》Area_Inf); free(p); p=temp; res = true; } } return res;}

看了上面的驅動函數,相信大家已經明白,大家可以自行編寫一些驅動,下面我實現的三個簡單函數。

/*** @brief 根據區域ID找到鏈表* @param data_p 鏈表指針* @param num 區域ID編號* @return int */int Find_Area_According_ID(Area_Inf_Typedef **data_p,int num){ Area_List_Inf_Typedef *p = Head_Area_List; while(p-》next_Area_Inf!=NULL) { p = p-》next_Area_Inf; if(p-》Area_Inf-》ID == num)//匹配到對應的值 { *data_p = p-》Area_Inf; return true; } } return false;}/*** @brief 刪除所有區域* */int Delete_All_Area(void){ int res = false; Area_List_Inf_Typedef *p = Head_Area_List; while(p-》next_Area_Inf!=NULL) { Area_List_Inf_Typedef *temp = p; p = p-》next_Area_Inf; temp-》next_Area_Inf = p-》next_Area_Inf; //釋放內存空間 free(p-》Area_Inf-》Area_Data); free(p-》Area_Inf); free(p); p=temp; res = true; } return res;}/*** @brief 打印鏈表信息* */void Printf_Area_Inf(void){ int i=0; Area_List_Inf_Typedef *p = Head_Area_List; printf(“list ID X Y Width Height Area_Data

”); while(p-》next_Area_Inf!=NULL) { p = p-》next_Area_Inf; printf(“ %d %d %d %d %d %d %s

”,i,p-》Area_Inf-》ID,p-》Area_Inf-》X,p-》Area_Inf-》Y,p-》Area_Inf-》Width,p-》Area_Inf-》Height,p-》Area_Inf-》Area_Data); i++; } printf(“----------------------end-----------------------

”);}

3、測試函數

下面編寫一個測試函數,可以測試,鏈表的初始化,增加一個區域,刪除指定區域,根據ID返回區域信息,刪除所有區域接口

/*** @brief 鏈表測試函數* */void list_main(){ int i,j; Area_Inf_Typedef temp; Area_Inf_Typedef **data_p; data_p = NULL; printf(“------------------List test---------------------

”); if(!Area_List_Init( )) { printf(“Memory fail.。

”); } for(i=0;i《5;i++) { temp.ID = i; temp.X = 5+i; temp.Y = i; temp.Width = 10+i; temp.Height = 10+i; temp.data_len = i+1; temp.Area_Data = (uint8_t*)malloc(temp.data_len+1); for(j=0;j《temp.data_len;j++) { temp.Area_Data[j] = j+0x30; } temp.Area_Data[j] = 0; if(!Add_Area_ToList(&temp)) { printf(“Add Area %d Area_Info fail

”,i); } } Printf_Area_Inf(); printf(“

-------------Delete ID of Area is 3-------------

”); Delete_Area_Accordingn_ID(3); Printf_Area_Inf(); temp.ID = 9; temp.data_len = 10; temp.Area_Data = (uint8_t*)malloc(temp.data_len+1); for(j=0;j《temp.data_len;j++) { temp.Area_Data[j] = j+0x30; } temp.Area_Data[j] = 0; if(!Add_Area_ToList(&temp)) { printf(“Add Area %d info fail

”,temp.ID); } printf(“

--------------Add ID of Area is 9---------------

”); Printf_Area_Inf(); Find_Area_According_ID(data_p,2); temp.ID = (*data_p)-》ID; Delete_All_Area(); printf(“

--------------Delete All Area-------------------

”); Printf_Area_Inf(); while(1);}

測試結果

6ef62022-9e2c-11eb-8b86-12bb97331649.png

IAR和keil工程代碼開源地址:

https://github.com/strongercjd/STM32_Linklist

(提示:公眾號不支持外鏈接,請復制鏈接到瀏覽器下載)

如果大家手中有板子可以調試,可以看《一文了解串口打印》文章,使用串口打印。如果臨時沒有板子可以debug,可以模擬測試,IAR設置如下:

選擇Simulator調試

6eff06a6-9e2c-11eb-8b86-12bb97331649.png

打開View-》TerminalI/O,就可以看到打印信息

6f0adc92-9e2c-11eb-8b86-12bb97331649.png

原文標題:鏈表在STM32中的應用

文章出處:【微信公眾號:玩轉單片機】歡迎添加關注!文章轉載請注明出處。

責任編輯:haq

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • STM32
    +關注

    關注

    2290

    文章

    11017

    瀏覽量

    362461
  • 鏈表
    +關注

    關注

    0

    文章

    80

    瀏覽量

    10790
收藏 人收藏

    評論

    相關推薦
    熱點推薦

    LTspice里壓敏電阻MOV怎么引入?

    LTspice里壓敏電阻MOV怎么引入
    發表于 04-28 08:26

    STM32微控制器實現數據加密的方法

    STM32微控制器實現數據加密,可以通過多種方法和技術來確保數據的安全性。以下是一些常見的方法和步驟: · 使用內置加密庫: · · STM32提供了專門的加密庫
    發表于 03-07 07:30

    AMC1200BDUBR工作時引入了112MHz和132MHz的EMC噪聲,怎么消除?

    請問前輩,我用的TI芯片:AMC1200BDUBR 工作時引入了112MHz和132MHz的EMC 噪聲。請問電路怎么改進才能消除這個噪聲?望前輩能幫忙推薦解決方法 不勝感激!
    發表于 12-23 07:31

    OpenAI探索新盈利模式:考慮AI產品引入廣告

    近日,據英國《金融時報》報道,OpenAI正在積極探索新的盈利模式,并考慮在其人工智能產品引入廣告。這一決策標志著OpenAI正在向盈利性企業轉型,試圖激烈的市場競爭開辟新的收入
    的頭像 發表于 12-03 11:05 ?579次閱讀

    HAL庫STM32開發的重要性

    HAL庫(Hardware Abstraction Layer Library,硬件抽象層庫)STM32開發扮演著至關重要的角色。以下是HAL庫
    的頭像 發表于 12-02 13:35 ?1186次閱讀

    索尼和本田將在新EV引入AI智駕功能

    近日,索尼集團與本田宣布了一項令人矚目的合作計劃,雙方將在新共同開發的純電動汽車(EV)引入人工智能(AI)自動駕駛輔助功能。這一舉措標志著日本車企首次公開引入該功能,為汽車行業帶來了新的變革。
    的頭像 發表于 10-22 17:12 ?627次閱讀

    放大電路引入反饋的作用

    反饋,簡而言之,就是將系統的輸出信號重新引入到輸入端的過程。放大電路,這通常意味著將輸出電壓或電流的一部分通過反饋網絡送回輸入端。這樣做的目的是利用輸出信號的信息來調節和控制輸入信號,進而
    的頭像 發表于 10-04 17:39 ?1226次閱讀
    放大電路<b class='flag-5'>中</b><b class='flag-5'>引入</b>反饋的作用

    AMC1200BDUBR工作時引入了112MHz和132MHz的EMC噪聲,請問電路怎么改進才能消除這個噪聲?

    請問前輩,我用的TI芯片:AMC1200BDUBR 工作時引入了112MHz和132MHz的EMC 噪聲。請問電路怎么改進才能消除這個噪聲?望前輩能幫忙推薦解決方法 不勝感激!
    發表于 08-22 06:26

    比較器轉換的過程為什么存在振蕩?

    。遲滯寬度?VT是根據輸入噪聲的大小以及振蕩的傾向來設定的。如圖2所示,將Vin和VR換接,就會形成一個具有遲滯特性的同相比較器。門限電壓會稍有不同。保證輸入信號是一致的。某些電路,輸出電平形成
    發表于 08-19 07:12

    AdobeIllustrator和Photoshop設計軟件引入新工具和生成性AI功能

    Adobe正在其Illustrator和Photoshop設計軟件引入一些新工具和生成性AI功能,旨在幫助加快創意工作流程。Illustrator最引人注目的更新來自Adobe最新的Firefly
    的頭像 發表于 07-29 17:10 ?1263次閱讀

    負反饋放大電路的作用

    放大電路引入負反饋,是一種廣泛采用的技術手段,旨在改善放大器的性能,提高輸出信號的質量,并使其更加穩定可靠。以下將從多個方面詳細闡述負反饋放大電路
    的頭像 發表于 07-22 18:10 ?3874次閱讀

    原定于iPhone系列引入樹脂涂覆銅箔計劃遭推遲

    7月18日,知名蘋果產品分析師郭明錤的最新報告揭示了蘋果公司一項重要決策的變動:原定于iPhone系列引入的新型樹脂涂覆銅箔(RCC)技術,其應用時間表再次遭遇調整。這項技術,曾預期
    的頭像 發表于 07-18 16:08 ?858次閱讀

    STM32H743怎么FreeRTOS移植canopen?

    STM32H743怎么FreeRTOS移植canopen?
    發表于 07-03 06:32

    如何在idf工程引入mdf WiFi-Mesh函數?

    我原先在idf下開發好的程序,如何引入mdf進行開發?需要用到WiFi-Mesh,看了下mdf下的例程是比較合適的,而idf下的wifi-mesh例程很粗略,想把mdf的例程移植到我原來的idf工程里面去
    發表于 06-28 14:59

    ESP32-S3的LCD接口可以用DMA鏈表來觸發發送數據嗎?

    因為是用來驅動LED顯示屏,用原來的I2S那樣并行,通過鏈接自己組織數據列表,還是比較方便的,現在S3的I2S好像已經不能并行發數據了,只能用LCD的接口了,所以想知道LCD接口的DMA能不能用鏈表來組織數據。
    發表于 06-17 07:25
    主站蜘蛛池模板: 亚洲成a人伦理 | 免费播放特黄特色毛片 | 无遮挡很爽很污很黄的网站w | 大象焦伊人久久综合网色视 | 素股中文字幕 | 国产视频黄色 | 97爱爱爱| 黄色网欧美 | 中文字幕天堂网 | 亚洲一级毛片免费看 | 人人插人人 | 日韩免费三级电影 | 婷婷六月综合 | 色欧美在线| 午夜视频在线观看免费高清 | 日本精品视频 | 美日毛片 | 亚洲一区二区三区免费在线观看 | 99精品在线 | 国模最新私拍视频在线观看 | 国产gav成人免费播放视频 | 午夜伦伦| 18岁禁黄色 | 在线播放网址 | 美女视频永久黄网站免费观看国产 | 亚洲插插插 | 午夜噜噜噜| 午夜视频在线免费播放 | 阿v视频在线观看免费播放 爱爱视频天天干 | sss在线play| 又粗又长又爽又长黄免费视频 | 五月婷婷视频在线观看 | 黄色大成网站 | 久久久久综合 | 91色欧美| 美女扒开下面让男人捅 | 日本特黄特色视频 | 天天干天天干天天色 | 超级乱淫小黄文小说 | 国产久爱青草视频在线观看 | www狠狠干 |