www.久久久久|狼友网站av天堂|精品国产无码a片|一级av色欲av|91在线播放视频|亚洲无码主播在线|国产精品草久在线|明星AV网站在线|污污内射久久一区|婷婷综合视频网站

當(dāng)前位置:首頁 > 單片機(jī) > C語言與CPP編程
[導(dǎo)讀]內(nèi)存池設(shè)計(jì)與實(shí)現(xiàn)一、前言作為C程序員,想必對于內(nèi)存操作這一塊是比較熟悉和操作比較頻繁的;比如申請一個對象,使用new,申請一塊內(nèi)存使用malloc等等;但是,往往會有一些困擾煩惱著大家,主要體現(xiàn)在兩部分:申請內(nèi)存后忘記釋放,造成內(nèi)存泄漏內(nèi)存不能循環(huán)使用,造成大量內(nèi)存碎片這兩個原因...

內(nèi)存池設(shè)計(jì)與實(shí)現(xiàn)

一、前言

作為C 程序員,想必對于內(nèi)存操作這一塊是比較熟悉和操作比較頻繁的;

比如申請一個對象,使用new,申請一塊內(nèi)存使用malloc等等;

但是,往往會有一些困擾煩惱著大家,主要體現(xiàn)在兩部分:

  • 申請內(nèi)存后忘記釋放,造成內(nèi)存泄漏
  • 內(nèi)存不能循環(huán)使用,造成大量內(nèi)存碎片
這兩個原因會影響我們程序長期平穩(wěn)的運(yùn)行,也有可能會導(dǎo)致程序的崩潰;

二、內(nèi)存池

內(nèi)存池是池化技術(shù)中的一種形式。通常我們在編寫程序的時候回使用 new delete 這些關(guān)鍵字來向操作系統(tǒng)申請內(nèi)存,而這樣造成的后果就是每次申請內(nèi)存和釋放內(nèi)存的時候,都需要和操作系統(tǒng)的系統(tǒng)調(diào)用打交道,從堆中分配所需的內(nèi)存。如果這樣的操作太過頻繁,就會找成大量的內(nèi)存碎片進(jìn)而降低內(nèi)存的分配性能,甚至出現(xiàn)內(nèi)存分配失敗的情況。

而內(nèi)存池就是為了解決這個問題而產(chǎn)生的一種技術(shù)。從內(nèi)存分配的概念上看,內(nèi)存申請無非就是向內(nèi)存分配方索要一個指針,當(dāng)向操作系統(tǒng)申請內(nèi)存時,操作系統(tǒng)需要進(jìn)行復(fù)雜的內(nèi)存管理調(diào)度之后,才能正確的分配出一個相應(yīng)的指針。而這個分配的過程中,我們還面臨著分配失敗的風(fēng)險(xiǎn)。

所以,每一次進(jìn)行內(nèi)存分配,就會消耗一次分配內(nèi)存的時間,設(shè)這個時間為 T,那么進(jìn)行 n 次分配總共消耗的時間就是 nT;如果我們一開始就確定好我們可能需要多少內(nèi)存,那么在最初的時候就分配好這樣的一塊內(nèi)存區(qū)域,當(dāng)我們需要內(nèi)存的時候,直接從這塊已經(jīng)分配好的內(nèi)存中使用即可,那么總共需要的分配時間僅僅只有 T。當(dāng) n 越大時,節(jié)約的時間就越多。

---引用來源互聯(lián)網(wǎng)

三、內(nèi)存池設(shè)計(jì)

內(nèi)存池設(shè)計(jì)實(shí)現(xiàn)中主要分為以下幾部分:

  • 重載new
  • 創(chuàng)建內(nèi)存節(jié)點(diǎn)
  • 創(chuàng)建內(nèi)存池
  • 管理內(nèi)存池
下面,比較詳細(xì)的來說說設(shè)計(jì)細(xì)節(jié):

重載new就不說了,直接從內(nèi)存節(jié)點(diǎn)開始;

內(nèi)存池節(jié)點(diǎn)

內(nèi)存池節(jié)點(diǎn)需要包含以下幾點(diǎn)元素:

  1. 所屬池子(pMem),因?yàn)楹罄m(xù)在內(nèi)存池管理中可以直接調(diào)用申請內(nèi)存和釋放內(nèi)存

  2. 下一個節(jié)點(diǎn)(pNext),這里主要是使用鏈表的思路,將所有的內(nèi)存塊關(guān)聯(lián)起來;

  3. 節(jié)點(diǎn)是否被使用(bUsed),這里保證每次使用前,該節(jié)點(diǎn)是沒有被使用的;

  4. 是否屬于內(nèi)存池(bBelong),主要是一般內(nèi)存池維護(hù)的空間都不是特別大,但是用戶申請了特別大的內(nèi)存時,就走正常的申請流程,釋放時也就正常釋放;

內(nèi)存池設(shè)計(jì)

內(nèi)存池設(shè)計(jì)就是上面的圖片類似,主要包含以下幾點(diǎn)元素:

  1. 內(nèi)存首地址(_pBuffer),也就是第一塊內(nèi)存,這樣以后方面尋找后面的內(nèi)存塊;
  2. 內(nèi)存塊頭(_pHeader),也就是上面說的內(nèi)存池節(jié)點(diǎn);
  3. 內(nèi)存塊大小(_nSize),也就是每個節(jié)點(diǎn)多大;
  4. 節(jié)點(diǎn)數(shù)(_nBlock),及時有多少個節(jié)點(diǎn);
這里面需要的注意的是,申請內(nèi)存塊的時候,需要加上節(jié)點(diǎn)頭,但是申請完后返回給客戶使用的需要去掉頭;但是釋放的時候,需要前移到頭,不然就會出現(xiàn)異常;

釋放內(nèi)存:

釋放內(nèi)存的時候,將使用過的內(nèi)存置為false,然后指向頭部,將頭部作為下一個節(jié)點(diǎn),這樣的話,節(jié)點(diǎn)每次回收就可以相應(yīng)的被找到;

內(nèi)存池管理

內(nèi)存池創(chuàng)建后,會根據(jù)節(jié)點(diǎn)大小和個數(shù)創(chuàng)建相應(yīng)的內(nèi)存池;

內(nèi)存池管理主要就是根據(jù)不同的需求創(chuàng)建不同的內(nèi)存池,以達(dá)到管理的目的;

這里主要有一個概念:數(shù)組映射

數(shù)組映射就是不同的范圍內(nèi),選擇不同的內(nèi)存池;

添一段代碼:

?void?InitArray(int?nBegin,int?nEnd,?MemoryPool*pMemPool)
?
{
??for?(int?i?=?nBegin;?i?<=?nEnd;?i )
??{
???_Alloc[i]?=?pMemPool;
??}
?}
根據(jù)范圍進(jìn)行綁定;

四、內(nèi)存池實(shí)現(xiàn)

ManagerPool.hpp

#ifndef?_MEMORYPOOL_HPP_
#define?_MEMORYPOOL_HPP_

#include?
#include?

////一個內(nèi)存塊的最大內(nèi)存大小,可以擴(kuò)展
#define?MAX_MEMORY_SIZE?256

class?MemoryPool;

//內(nèi)存塊
struct?MemoryBlock
{

?MemoryBlock*?pNext;//下一塊內(nèi)存塊
?bool?bUsed;//是否使用
?bool?bBelong;//是否屬于內(nèi)存池
?MemoryPool*?pMem;//屬于哪個池子
};

class?MemoryPool
{

public:
?MemoryPool(size_t?nSize=128,size_t?nBlock=10)
?{
??//相當(dāng)于申請10塊內(nèi)存,每塊內(nèi)存是1024
??_nSize?=?nSize;
??_nBlock?=?nBlock;
??_pHeader?=?NULL;
??_pBuffer?=?NULL;
?}
?virtual?~MemoryPool()
?{
??if?(_pBuffer?!=?NULL)
??{
???free(_pBuffer);
??}
?}
?//申請內(nèi)存
?void*?AllocMemory(size_t?nSize)
?
{
??std::lock_guard<std::mutex>?lock(_mutex);
??//如果首地址為空,說明沒有申請空間
??if?(_pBuffer?==?NULL)
??{
???InitMemory();
??}
??MemoryBlock*?pRes?=?NULL;
??//如果內(nèi)存池不夠用時,需要重新申請內(nèi)存
??if?(_pHeader?==?NULL)
??{
???pRes?=?(MemoryBlock*)malloc(nSize sizeof(MemoryBlock));
???pRes->bBelong?=?false;
???pRes->bUsed?=?false;
???pRes->pNext?=?NULL;
???pRes->pMem?=?NULL;
??}
??else
??{
???pRes?=?_pHeader;
???_pHeader?=?_pHeader->pNext;
???pRes->bUsed?=?true;
??}
??//返回只返回頭后面的信息
??return?((char*)pRes? ?sizeof(MemoryBlock));
?}

?//釋放內(nèi)存
?void?FreeMemory(void*?p)
?
{
??std::lock_guard<std::mutex>?lock(_mutex);
??//和申請內(nèi)存剛好相反,這里需要包含頭,然后全部釋放
??MemoryBlock*?pBlock?=?((MemoryBlock*)p?-?sizeof(MemoryBlock));
??if?(pBlock->bBelong)
??{
???pBlock->bUsed?=?false;
???//循環(huán)鏈起來
???pBlock->pNext?=?_pHeader;
???pBlock?=?_pHeader;
??}
??else
??{
???//不屬于內(nèi)存池直接釋放就可以
???free(pBlock);
??}
?}
?//初始化內(nèi)存塊
?void?InitMemory()
?
{
??if?(_pBuffer)
???return;
??//計(jì)算每塊的大小
??size_t?PoolSize?=?_nSize? ?sizeof(MemoryBlock);
??//計(jì)算需要申請多少內(nèi)存
??size_t?BuffSize?=?PoolSize?*?_nBlock;
??_pBuffer?=?(char*)malloc(BuffSize);
??//初始化頭
??_pHeader?=?(MemoryBlock*)_pBuffer;
??_pHeader->bUsed?=?false;
??_pHeader->bBelong?=?true;
??_pHeader->pMem?=?this;
??//初始化_nBlock塊,并且用鏈表的形式連接
??//保存頭指針
??MemoryBlock*?tmp1?=?_pHeader;
??for?(size_t?i?=?1;?i???{
???MemoryBlock*?tmp2?=?(MemoryBlock*)(_pBuffer? ?i*PoolSize);
???tmp2->bUsed?=?false;
???tmp2->pNext?=?NULL;
???tmp2->bBelong?=?true;
???_pHeader->pMem?=?this;
???tmp1->pNext?=?tmp2;
???tmp1?=?tmp2;
??}
?}
public:
?//內(nèi)存首地址(第一塊內(nèi)存的地址)
?char*?_pBuffer;
?//內(nèi)存塊頭
?MemoryBlock*?_pHeader;
?//內(nèi)存塊大小
?size_t?_nSize;
?//多少塊
?size_t?_nBlock;

?std::mutex?_mutex;
};

//可以使用模板傳遞參數(shù)
template<size_t?nSize,size_t?nBlock>
class?MemoryPoolor:public?MemoryPool
{
public:
?MemoryPoolor()
?{
??_nSize?=?nSize;
??_nBlock?=?nBlock;
?}

};

//需要重新對內(nèi)存池就行管理
class?ManagerPool
{

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

LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: 驅(qū)動電源

在工業(yè)自動化蓬勃發(fā)展的當(dāng)下,工業(yè)電機(jī)作為核心動力設(shè)備,其驅(qū)動電源的性能直接關(guān)系到整個系統(tǒng)的穩(wěn)定性和可靠性。其中,反電動勢抑制與過流保護(hù)是驅(qū)動電源設(shè)計(jì)中至關(guān)重要的兩個環(huán)節(jié),集成化方案的設(shè)計(jì)成為提升電機(jī)驅(qū)動性能的關(guān)鍵。

關(guān)鍵字: 工業(yè)電機(jī) 驅(qū)動電源

LED 驅(qū)動電源作為 LED 照明系統(tǒng)的 “心臟”,其穩(wěn)定性直接決定了整個照明設(shè)備的使用壽命。然而,在實(shí)際應(yīng)用中,LED 驅(qū)動電源易損壞的問題卻十分常見,不僅增加了維護(hù)成本,還影響了用戶體驗(yàn)。要解決這一問題,需從設(shè)計(jì)、生...

關(guān)鍵字: 驅(qū)動電源 照明系統(tǒng) 散熱

根據(jù)LED驅(qū)動電源的公式,電感內(nèi)電流波動大小和電感值成反比,輸出紋波和輸出電容值成反比。所以加大電感值和輸出電容值可以減小紋波。

關(guān)鍵字: LED 設(shè)計(jì) 驅(qū)動電源

電動汽車(EV)作為新能源汽車的重要代表,正逐漸成為全球汽車產(chǎn)業(yè)的重要發(fā)展方向。電動汽車的核心技術(shù)之一是電機(jī)驅(qū)動控制系統(tǒng),而絕緣柵雙極型晶體管(IGBT)作為電機(jī)驅(qū)動系統(tǒng)中的關(guān)鍵元件,其性能直接影響到電動汽車的動力性能和...

關(guān)鍵字: 電動汽車 新能源 驅(qū)動電源

在現(xiàn)代城市建設(shè)中,街道及停車場照明作為基礎(chǔ)設(shè)施的重要組成部分,其質(zhì)量和效率直接關(guān)系到城市的公共安全、居民生活質(zhì)量和能源利用效率。隨著科技的進(jìn)步,高亮度白光發(fā)光二極管(LED)因其獨(dú)特的優(yōu)勢逐漸取代傳統(tǒng)光源,成為大功率區(qū)域...

關(guān)鍵字: 發(fā)光二極管 驅(qū)動電源 LED

LED通用照明設(shè)計(jì)工程師會遇到許多挑戰(zhàn),如功率密度、功率因數(shù)校正(PFC)、空間受限和可靠性等。

關(guān)鍵字: LED 驅(qū)動電源 功率因數(shù)校正

在LED照明技術(shù)日益普及的今天,LED驅(qū)動電源的電磁干擾(EMI)問題成為了一個不可忽視的挑戰(zhàn)。電磁干擾不僅會影響LED燈具的正常工作,還可能對周圍電子設(shè)備造成不利影響,甚至引發(fā)系統(tǒng)故障。因此,采取有效的硬件措施來解決L...

關(guān)鍵字: LED照明技術(shù) 電磁干擾 驅(qū)動電源

開關(guān)電源具有效率高的特性,而且開關(guān)電源的變壓器體積比串聯(lián)穩(wěn)壓型電源的要小得多,電源電路比較整潔,整機(jī)重量也有所下降,所以,現(xiàn)在的LED驅(qū)動電源

關(guān)鍵字: LED 驅(qū)動電源 開關(guān)電源

LED驅(qū)動電源是把電源供應(yīng)轉(zhuǎn)換為特定的電壓電流以驅(qū)動LED發(fā)光的電壓轉(zhuǎn)換器,通常情況下:LED驅(qū)動電源的輸入包括高壓工頻交流(即市電)、低壓直流、高壓直流、低壓高頻交流(如電子變壓器的輸出)等。

關(guān)鍵字: LED 隧道燈 驅(qū)動電源
關(guān)閉