RM新时代网站-首页

0
  • 聊天消息
  • 系統(tǒng)消息
  • 評論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

干貨 | 嵌入式C語言的內(nèi)存管理

AGk5_ZLG_zhiyua ? 來源:YXQ ? 2019-07-23 14:32 ? 次閱讀

很多工程師都知道,C/C++語言與其他語言不同,它需要開發(fā)者自己管理內(nèi)存資源,動(dòng)態(tài)內(nèi)存使用不當(dāng),容易造成段錯(cuò)誤或者內(nèi)存泄漏,因此內(nèi)存管理至關(guān)重要。本文將以C語言為例介紹動(dòng)態(tài)內(nèi)存管理的原理。

C/C++語言與其他語言不同,它需要開發(fā)者自己管理內(nèi)存資源。對于動(dòng)態(tài)內(nèi)存的使用不當(dāng)容易造成段錯(cuò)誤或者內(nèi)存泄漏。尤其是內(nèi)存泄漏,內(nèi)存泄漏往往是在程序運(yùn)行一段時(shí)間才會(huì)被發(fā)現(xiàn),使得開發(fā)人員無法第一時(shí)間定位錯(cuò)誤。

而相比于個(gè)人計(jì)算機(jī),嵌入式系統(tǒng)的內(nèi)存資源更是稀缺。作為嵌入式C的開發(fā)人員,了解其內(nèi)存管理的原理能使其更加正確地使用內(nèi)存資源以及定位程序的bug。

動(dòng)態(tài)內(nèi)存的原理

1、棧空間與堆空間

在介紹內(nèi)存管理之前,我們先解釋一下棧空間與堆空間:??臻g是由編譯器自動(dòng)分配釋放,對于AWorks等操作系統(tǒng),在用戶創(chuàng)建一個(gè)任務(wù)的時(shí)候可以由自己決定任務(wù)棧空間的大小。

棧空間里面一般存放著如下數(shù)據(jù):在函數(shù)內(nèi)的局部變量(不包括static定義的變量),在調(diào)用另一個(gè)函數(shù)時(shí)保存的通用寄存器信息等。

參考如下例程(為了便于理解,省略通用寄存器等信息):

在task執(zhí)行s = calculate_sum(a,b);之前,task的棧內(nèi)保存如下數(shù)據(jù):

程序接下來執(zhí)行calculate_sum函數(shù),其棧向下增長。在返回task之前,其棧結(jié)構(gòu)如下:

執(zhí)行完calculate_sum之后,根據(jù)返回地址返回task之后,?;謴?fù)調(diào)用之前的結(jié)構(gòu):

所以??臻g存儲(chǔ)著代碼塊內(nèi)的局部變量,動(dòng)態(tài)地增減著內(nèi)部的數(shù)據(jù)。這也就是為什么當(dāng)接口調(diào)用結(jié)束后變量就不再“生存”的原因。

堆空間是由OS管理的一片區(qū)域,開發(fā)者可向OS動(dòng)態(tài)申請一片區(qū)域用于操作數(shù)據(jù)。

堆空間在程序運(yùn)行時(shí)一直有效,相當(dāng)于定義了一個(gè)大型的全局?jǐn)?shù)組。需要時(shí)向堆空間申請內(nèi)存,使用完畢再還回去。這樣可以使得開發(fā)者能夠動(dòng)態(tài)地控制空間的大小,而不需要在寫代碼的時(shí)候就考慮最糟的情況(定義一個(gè)數(shù)組必須在編譯之前就確定其大小,使用過程中無法增加或減少,所以必須考慮最多需要的數(shù)據(jù)大?。?。

堆空間由編譯器決定,如果開發(fā)者想嘗試實(shí)現(xiàn)一片動(dòng)態(tài)內(nèi)存,可向堆申請一片對齊的內(nèi)存空間。

2、內(nèi)存資源的申請與釋放

我們這里以常用的內(nèi)存操作接口——malloc與free為例,介紹操作動(dòng)態(tài)內(nèi)存的細(xì)節(jié)。

void* malloc(size)——申請一片大小為size字節(jié)的內(nèi)存。

參考下圖,灰色部分是已經(jīng)被使用的內(nèi)存,空白部分則是可以被申請使用的內(nèi)存。在申請內(nèi)存的時(shí)候,系統(tǒng)會(huì)首先判斷有沒有足夠大的未被使用的區(qū)域,如果有,則將其分配給申請者,再將此區(qū)域標(biāo)記為“已使用”;否則分配失敗。

(為方便讀圖,從這里開始我們假定內(nèi)存的地址從上往下增長)

void free(void *)——釋放已申請的內(nèi)存。與malloc相反,free的作用是把“已使用”的區(qū)域標(biāo)記為“未使用”,那么釋放的內(nèi)存下一次就可以再分配出去復(fù)用。free釋放的內(nèi)存必須是malloc申請的內(nèi)存。

由于需要對內(nèi)存進(jìn)行狀態(tài)標(biāo)記和位置記錄(以便釋放)。在申請/釋放內(nèi)存的時(shí)候需要額外的空間進(jìn)行信息的記錄。有的系統(tǒng)會(huì)將記錄的信息集中管理,有的則是申請內(nèi)存的時(shí)候額外地多申請一小片區(qū)域用于記錄。

3、內(nèi)存泄漏

對于動(dòng)態(tài)申請的內(nèi)存,使用完畢之后應(yīng)該還給堆,才能在后續(xù)繼續(xù)分配出去。而如果申請的內(nèi)存如果沒有還回去,就造成了內(nèi)存泄漏。參考如下一段代碼:

現(xiàn)在我們設(shè)flag=1,執(zhí)行這個(gè)函數(shù)會(huì)發(fā)生什么?

首先ptr會(huì)指向申請的128字節(jié)的內(nèi)存(圖b),然后判斷flag==1之后再申請256字節(jié)的內(nèi)存(圖c)。假設(shè)我們現(xiàn)在使用完畢將ptr釋放:

現(xiàn)在我們釋放了256字節(jié)的內(nèi)存塊了,但是我們開始的時(shí)候還申請過128字節(jié)的內(nèi)存塊,這128字節(jié)的內(nèi)存塊最終會(huì)怎樣呢?由當(dāng)時(shí)唯一指向這塊內(nèi)存的指針ptr后面指向了256字節(jié)的內(nèi)存塊,現(xiàn)在沒有任何指針指向這塊內(nèi)存,因此這一塊內(nèi)存再也無法被釋放,這時(shí)候我們就說內(nèi)存泄漏了。

在程序最開始運(yùn)行的一段時(shí)間內(nèi),系統(tǒng)是沒有異常的。即使一小片內(nèi)存不被釋放也不會(huì)造成錯(cuò)誤,因?yàn)閮?nèi)存堆還有足夠的空間可以使用。但是如果運(yùn)行的時(shí)間足夠長,多次調(diào)用這個(gè)函數(shù)(參數(shù)flag==1)之后,堆空間會(huì)逐漸被泄漏的內(nèi)存塊占滿,直到程序無法再從堆里申請到內(nèi)存,程序才會(huì)報(bào)錯(cuò)。

內(nèi)存泄漏令開發(fā)者頭痛的地方也正是這個(gè)原因,內(nèi)存泄漏的問題往往無法在第一時(shí)間被發(fā)現(xiàn)!而對于不熟悉內(nèi)存管理的開發(fā)者更是難以定位錯(cuò)誤。

對于動(dòng)態(tài)內(nèi)存的操作,需要時(shí)刻記?。寒?dāng)一塊申請的內(nèi)存不再使用的時(shí)候,必須及時(shí)釋放。一個(gè)malloc操作需要對應(yīng)一個(gè)free操作。

4、內(nèi)存對齊

在很多的場合下,分配的內(nèi)存不僅要滿足申請的大小,也需要進(jìn)行對齊才能夠使分配的空間能夠被轉(zhuǎn)換成除char之外其他的結(jié)構(gòu)類型。系統(tǒng)對內(nèi)存的分配一般會(huì)以int型變量的字節(jié)數(shù)進(jìn)行對齊。AWorks提供的aw_mem_align接口可以使用戶獲取自定義對齊的內(nèi)存空間。

類型對齊相關(guān)的知識請讀者自行查找相關(guān)資料,這里不再展開細(xì)講。

內(nèi)存管理算法

接下來我們學(xué)習(xí)一下怎么去對堆空間的內(nèi)存進(jìn)行管理。這里我們主要介紹嵌入式中兩種常用的內(nèi)存管理算法。

1、鏈表法

鏈表法維護(hù)著兩個(gè)鏈表,兩個(gè)鏈表分別記錄著已分配的使用內(nèi)存段和未分配的空閑內(nèi)存段。當(dāng)申請一片內(nèi)存的時(shí)候,從空閑內(nèi)存段中找到合適的塊分配給用戶,同時(shí)鏈入使用內(nèi)存段。而釋放的時(shí)候則從已使用的內(nèi)存段找到相應(yīng)的表,然后釋放到空閑內(nèi)存段中以供下次分配。

現(xiàn)在我們以最先匹配算法為例介紹算法的細(xì)節(jié)。最先匹配算法是從空閑鏈表的表頭出發(fā),依次尋找空閑的內(nèi)存,一旦找到足夠大的連續(xù)區(qū)域則將其返回給用戶。

還記得前面說的,管理內(nèi)存區(qū)域需要額外的記錄信息,鏈表法一般是在操作內(nèi)存空間的時(shí)候申請額外的空間記錄相應(yīng)的信息。我們假定下圖紅色部分記錄著使用的內(nèi)存區(qū),青色記錄空閑的內(nèi)存區(qū),這里使用free link鏈表維護(hù)空閑內(nèi)存段,used link維護(hù)使用的內(nèi)存段:

程序運(yùn)行一段時(shí)間之后,假設(shè)堆內(nèi)的空間分布如下:

空閑區(qū)和使用區(qū)的信息都被兩個(gè)表維護(hù)著。

現(xiàn)在用戶需要申請一片大小為3k的內(nèi)存,系統(tǒng)會(huì)從free link出發(fā),先是找到2k的空閑區(qū),由于2k的空間不夠用,接下來再繼續(xù)尋找,找到了4k的區(qū)域,發(fā)現(xiàn)4k的區(qū)域夠大了,就會(huì)將4k的空間取走3k的空間并將其鏈入used link。

盡管后面3k的空間更加適合分配,但是最先匹配算法一旦找到足夠大的空間便不會(huì)繼續(xù)往下尋找。

當(dāng)用戶用完資源的時(shí)候,把申請的3k還回去,系統(tǒng)會(huì)從used link找到申請的內(nèi)存,將鏈入free link以供下次分配,然后將空閑相鄰的內(nèi)存塊合并成完整的一塊:

現(xiàn)在考慮這樣的一種情況:假設(shè)用戶要申請5k的內(nèi)存塊,系統(tǒng)能夠提供嗎?并不能。

雖然空閑的內(nèi)存塊一共有9k(2k+4k+3k),但是9k的內(nèi)存并不連續(xù),因此無法分配給用戶。這就是外部內(nèi)存碎片——雖然整個(gè)空間的空閑內(nèi)存足夠大,但卻因?yàn)榱闼榈膬?nèi)存塊割裂了連續(xù)內(nèi)存而無法分配出去。

其他的鏈表法還有最佳匹配算法,下次匹配算法等.有興趣的讀者可以自行查找相關(guān)資料。

2、位圖法

使用位圖法,系統(tǒng)的內(nèi)存會(huì)被劃分成固定的內(nèi)存塊。再用變量的其中一位指示其中的一塊內(nèi)存:

圖中的一個(gè)方格代表一塊固定大小的內(nèi)存塊,這里假定1k。用一個(gè)16位的變量指代16k的內(nèi)存段。如果一個(gè)塊是空閑的,則用0表示,如果是被使用的,則用1表示。

下圖的第1,2個(gè)內(nèi)存塊和第7,8,9個(gè)內(nèi)存塊都被使用了,而相應(yīng)的位都被置1說明被占用了。

相比鏈表法,位圖法采用更少的額外空間記錄內(nèi)存堆的信息,而且由于申請與釋放都是整塊的,會(huì)產(chǎn)生更少的外部碎片。

但是假如用戶只申請幾個(gè)字節(jié)的內(nèi)存,但是卻分配了1k的內(nèi)存塊,則大量的空間不會(huì)被使用,這樣導(dǎo)致的無法使用的內(nèi)存我們稱為內(nèi)部內(nèi)存碎片。

減少內(nèi)部內(nèi)存碎片的其中一個(gè)方法是合理地選擇內(nèi)存塊的大小,固定尺寸較小的內(nèi)存塊導(dǎo)致的內(nèi)存碎片會(huì)更小——當(dāng)用戶申請幾字節(jié)的內(nèi)存,比起固定1k的內(nèi)存塊,固定16字節(jié)的內(nèi)存塊產(chǎn)生更少的碎片。但是固定尺寸變小了也會(huì)導(dǎo)致需要更多的位記錄內(nèi)存的信息。

現(xiàn)在有很多優(yōu)秀的位圖算法——將內(nèi)存分成不同的固定大小獲取更快的分配速度和更少的內(nèi)存碎片,有興趣的讀者可自行查找相關(guān)資料。

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請聯(lián)系本站處理。 舉報(bào)投訴
  • 嵌入式
    +關(guān)注

    關(guān)注

    5082

    文章

    19104

    瀏覽量

    304799
  • C語言
    +關(guān)注

    關(guān)注

    180

    文章

    7604

    瀏覽量

    136685

原文標(biāo)題:AWorks編程:嵌入式C語言的內(nèi)存管理

文章出處:【微信號:ZLG_zhiyuan,微信公眾號:ZLG致遠(yuǎn)電子】歡迎添加關(guān)注!文章轉(zhuǎn)載請注明出處。

收藏 人收藏

    評論

    相關(guān)推薦

    嵌入式 C 語言

    庫函數(shù)支撐調(diào)用,分配的內(nèi)存是電腦的內(nèi)存,其處理器就是電腦的CPU;而在嵌入式環(huán)境中,會(huì)涉及到底層的硬件,而硬件本身是沒有標(biāo)準(zhǔn)庫可以調(diào)用的,因而就需要開發(fā)者使用C
    發(fā)表于 07-01 08:58

    標(biāo)準(zhǔn)C語言嵌入式C語言有哪些區(qū)別

    嵌入式系統(tǒng)是各行各業(yè)的具體應(yīng)用相結(jié)合的產(chǎn)物。例如計(jì)算機(jī)技術(shù)、半導(dǎo)體技術(shù)、電子技術(shù)。其更在乎效率和內(nèi)存有效使用。嵌入式系統(tǒng)是一個(gè)技術(shù)密集、資金密集、高度分散、不斷創(chuàng)新的知識集成系統(tǒng)。C
    發(fā)表于 12-14 06:15

    如何使用嵌入式內(nèi)存分配管理技術(shù)

    嵌入式---內(nèi)存分配管理嵌入式內(nèi)存一般都非常的小,最進(jìn)在學(xué)習(xí)LWIP協(xié)議棧的移植,在正點(diǎn)原子的學(xué)習(xí)資料中找到了許多關(guān)于怎么移植協(xié)議棧的東西
    發(fā)表于 12-17 06:41

    ARM嵌入式系統(tǒng)C語言編程

    無操作系統(tǒng)支持的嵌入式系統(tǒng)軟件,包括系統(tǒng)引導(dǎo)(BOOT) 、驅(qū)動(dòng)程序、動(dòng)態(tài)內(nèi)存管理、IPO、通信以及應(yīng)用軟件等方面。本文詳細(xì)介紹了嵌入式平臺上用C
    發(fā)表于 11-07 15:55 ?165次下載

    嵌入式c語言編程(由淺入深)

    本內(nèi)容詳細(xì)介紹了嵌入式c語言編程的各項(xiàng)知識,包括嵌入式c語言編程,
    發(fā)表于 11-02 14:37 ?0次下載
    <b class='flag-5'>嵌入式</b><b class='flag-5'>c</b><b class='flag-5'>語言</b>編程(由淺入深)

    嵌入式外中斷c語言代碼

    嵌入式外中斷c語言代碼(arm嵌入式開發(fā)實(shí)例)-嵌入式外中斷c
    發(fā)表于 07-30 11:29 ?4次下載
    <b class='flag-5'>嵌入式</b>外中斷<b class='flag-5'>c</b><b class='flag-5'>語言</b>代碼

    嵌入式C語言-文件操用

    嵌入式C語言-文件操用(嵌入式開發(fā)需要什么證書)-嵌入式C
    發(fā)表于 07-30 11:56 ?15次下載
    <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>-文件操用

    標(biāo)準(zhǔn)c語言嵌入式,嵌入式C語言C語言的區(qū)別

    嵌入式C語言C語言的區(qū)別:最常用的系統(tǒng)編程語言C
    發(fā)表于 10-20 14:06 ?6次下載
    標(biāo)準(zhǔn)<b class='flag-5'>c</b><b class='flag-5'>語言</b>與<b class='flag-5'>嵌入式</b>,<b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>與<b class='flag-5'>C</b><b class='flag-5'>語言</b>的區(qū)別

    嵌入式linux c語言,嵌入式LinuxC語言開發(fā)工具.pdf

    2 章 嵌入式Linux C 語言開發(fā)工具本章目標(biāo)任何應(yīng)用程序的開發(fā)都離不開編輯器、編譯器及調(diào)試器,嵌入式Linux 的C
    發(fā)表于 11-01 17:38 ?12次下載
    <b class='flag-5'>嵌入式</b>linux <b class='flag-5'>c</b><b class='flag-5'>語言</b>,<b class='flag-5'>嵌入式</b>LinuxC<b class='flag-5'>語言</b>開發(fā)工具.pdf

    嵌入式 Linux 中的內(nèi)存管理

    點(diǎn)擊 嵌入式 Linux 中的內(nèi)存管理
    發(fā)表于 11-02 10:36 ?12次下載
    <b class='flag-5'>嵌入式</b> Linux 中的<b class='flag-5'>內(nèi)存</b><b class='flag-5'>管理</b>

    C語言嵌入式培訓(xùn) 嵌入式C語言程序設(shè)計(jì)基礎(chǔ)

      學(xué)習(xí)嵌入式的基礎(chǔ)語言C語言,因此先掌握C語言對于后續(xù)
    發(fā)表于 11-03 21:06 ?32次下載
    <b class='flag-5'>C</b><b class='flag-5'>語言</b><b class='flag-5'>嵌入式</b>培訓(xùn)  <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>程序設(shè)計(jì)基礎(chǔ)

    嵌入式C語言知識總結(jié)

    1 嵌入式C語言總結(jié)從語法上來說C語言并不復(fù)雜, 但編寫優(yōu)質(zhì)可靠的嵌入式
    發(fā)表于 12-20 19:44 ?12次下載
    <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>知識總結(jié)

    c語言嵌入式編程

    比較詳盡的嵌入式C語言解答和分析
    發(fā)表于 03-10 14:53 ?162次下載

    嵌入式C語言的結(jié)構(gòu)特點(diǎn)

    嵌入式開發(fā)中既有底層硬件的開發(fā)又涉及上層應(yīng)用的開發(fā),即涉及系統(tǒng)的硬件和軟件,C語言既具有匯編語言操作底層的優(yōu)勢,又具有高級語言功能性強(qiáng)的特點(diǎn)
    的頭像 發(fā)表于 11-24 16:16 ?671次閱讀
    <b class='flag-5'>嵌入式</b><b class='flag-5'>C</b><b class='flag-5'>語言</b>的結(jié)構(gòu)特點(diǎn)

    嵌入式C語言高手煉成之內(nèi)存操作篇

    嵌入式系統(tǒng)的編程中,常常要求在特定的內(nèi)存單元讀寫內(nèi)容,匯編有對應(yīng)的MOV指令,而除C/C++以外的其它編程語言基本沒有直接訪問絕對地址的能
    的頭像 發(fā)表于 12-11 17:20 ?509次閱讀
    RM新时代网站-首页