您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“C標(biāo)準(zhǔn)庫堆內(nèi)存函數(shù)的示例分析”,內(nèi)容簡而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“C標(biāo)準(zhǔn)庫堆內(nèi)存函數(shù)的示例分析”這篇文章吧。
C標(biāo)準(zhǔn)庫堆內(nèi)存函數(shù)有4個:malloc、free、calloc、realloc,其函數(shù)聲明放在了#include <stdlib.h>中,主要用來申請和釋放堆內(nèi)存。
堆內(nèi)存的申請和釋放(wiki,chs),需要發(fā)起系統(tǒng)調(diào)用,會帶來昂貴的上下文切換(用戶態(tài)切換到內(nèi)核態(tài)),十分耗時。另外,這些過程可能是帶鎖的,難以并行化。
對于操作系統(tǒng)而言,內(nèi)存管理的基本單位是頁(通常為4K),而不是需要4 Bytes時,就給你分配4 Bytes,釋放4 Bytes時,就給你釋放4 Bytes。
因此,為了提升效率,操作系統(tǒng)會調(diào)用系統(tǒng)api(windows上是VirtualAlloc、VirtualFree,其他平臺是mmap、munmap)來實現(xiàn)一個Ansi C內(nèi)存分配器(工作在用戶態(tài)),供C標(biāo)準(zhǔn)庫堆內(nèi)存函數(shù)來使用。
windows:MSVCRT.DLL中使用NT heap實現(xiàn)
linux:glibc中使用ptmalloc實現(xiàn)
Android:使用jemalloc實現(xiàn)
(1)內(nèi)存碎片(fragmentation):即空閑內(nèi)存不能被利用。分為外部碎片(在分配單元間的未使用的內(nèi)存);內(nèi)部碎片(在分配單元中未使用的內(nèi)存)
(2)內(nèi)存碎片的罪魁禍?zhǔn)拙褪切K內(nèi)存的頻繁分配
(3)內(nèi)存碎片無法避免,只能通過內(nèi)存分配器算法來減少,例如:接近大小的內(nèi)存就近分配,釋放時能合并就合并,從而減少碎片
(4)上面講的內(nèi)存碎片指的是虛擬內(nèi)存碎片,OS是不管的,OS只管物理內(nèi)存。
(5)平時我們說的內(nèi)存碎片整理(defragment)或內(nèi)存緊縮(memory compaction),是指OS對物理內(nèi)存進(jìn)行的碎片整理,把分開小的物理內(nèi)存頁移動在一起形成一個大的整塊。
OS整理完物理內(nèi)存后,會用新的物理內(nèi)存地址來更新虛擬內(nèi)存與物理內(nèi)存映射表,這些對于上層邏輯都是透明的。
虛擬內(nèi)存是不能進(jìn)行碎片整理的,主要原因是碎片整理會移動內(nèi)存,上層邏輯的指針地址確還是指向老的地址,這會導(dǎo)致致命錯誤。
(1)分配和釋放的效率
(2)內(nèi)存分配器的利用率。包括以下幾個方面:
① 內(nèi)存對齊導(dǎo)致的不可使用的內(nèi)存碎片(內(nèi)部碎片)
② 內(nèi)存碎片太嚴(yán)重,使得分配大塊內(nèi)存時,找不到空閑塊,最后導(dǎo)致內(nèi)存分配失?。ㄍ獠克槠?/p>
③ 內(nèi)存頁始終有被使用,導(dǎo)致分配器無法及時釋放該頁的內(nèi)存占用,使得整個內(nèi)存分配器的內(nèi)存占用被撐得很大,縮不回去
形參size為要求分配的字節(jié)數(shù)。如果函數(shù)執(zhí)行成功,malloc返回獲得內(nèi)存空間的首地址;如果函數(shù)執(zhí)行失敗,那么返回值為NULL。
由于 malloc函數(shù)值的類型為void型指針,因此,可以將其值類型轉(zhuǎn)換后賦給任意類型指針,這樣就可以通過操作該類型指針來操作從堆上獲得的內(nèi)存空間。
需要注意的是,malloc函數(shù)分配得到的內(nèi)存空間是未初始化的。可通過調(diào)用memset來將其初始化為全0。
int* p = (int *) malloc(sizeof(int)*100); if (p == NULL) { printf("Can't get memory!\n"); } memset(p, 0, sizeof(int)*100);
從堆上獲得的內(nèi)存,在程序結(jié)束之前,系統(tǒng)不會將其自動釋放,需要程序員來自己管理,防止出現(xiàn)內(nèi)存泄露。
free(p); p = NULL;
calloc函數(shù)的功能與malloc函數(shù)的功能相似,都是從堆分配內(nèi)存。
函數(shù)返回值為void*。如果執(zhí)行成功,從堆上獲得size * num大小的堆內(nèi)存,并返回該內(nèi)存塊的首地址。如果執(zhí)行失敗,函數(shù)返回NULL。
與malloc函數(shù)不同的是,calloc函數(shù)得到的內(nèi)存塊會被初始化為全0。由于提供了2個參數(shù),比較適合為數(shù)組申請空間,可以將size設(shè)置為數(shù)組元素的空間長度,將num設(shè)置為數(shù)組的容量。
int* p = (int *) calloc(100, sizeof(int)); if (p == NULL) { printf("Can't get memory!\n"); }
為ptr重新分配大小為size的一塊內(nèi)存空間。下面是這個函數(shù)的工作流程:
① 如果ptr為NULL,則函數(shù)相當(dāng)于malloc(new_size),試著分配一塊大小為new_size的內(nèi)存,如果成功將地址返回,否則返回NULL。
② 如果ptr不為NULL,查看ptr是不是在堆中,如果不是的話會拋出realloc invalid pointer異常。如果ptr在堆中,則查看new_size大小。
(a)如果new_size大小為0,則相當(dāng)于free(ptr),將ptr指向的內(nèi)存空間釋放掉,返回NULL。
(b)如果new_size小于原大小,只有new_size大小的數(shù)據(jù)會保存,后面地址的數(shù)據(jù)可能會丟失;
(c)如果new_size等于原大小,什么都沒有做;
(d)如果new_size大于原大小,則查看ptr指向的位置還有沒有足夠的連續(xù)內(nèi)存空間,如果有的話,分配更多的空間,返回的地址和ptr相同;
如果沒有的話,在更大的空間內(nèi)查找,如果找到new_size大小的空間,將舊的內(nèi)容拷貝到新的內(nèi)存中,把舊的內(nèi)存釋放掉,則返回新地址,否則返回NULL。
int* p = (int*)malloc(sizeof(int)); *p = 3; printf("p=%p\n", p); // p=0000020B2966E310 printf("*p=%d\n", *p); // *p=3 p = (int*)realloc(p, sizeof(int)); // 什么也不做 printf("p=%p\n", p); // p=0000020B2966E310 printf("*p=%d\n", *p); // *p=3 p = (int*)realloc(p, 1024 * sizeof(int)); // 創(chuàng)建4KB的內(nèi)存塊 注:4KB為一個頁面的大小 printf("p=%p\n", p); // p=0000020B29673A50 注:由于不能在原來地址上擴(kuò)容,會將原來地址內(nèi)存釋放,并在新地址申請內(nèi)存塊 printf("*p=%d\n", *p); // *p=3 realloc(p, 0); // 相當(dāng)于free(p) p = NULL;
以上是“C標(biāo)準(zhǔn)庫堆內(nèi)存函數(shù)的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對大家有所幫助,如果還想學(xué)習(xí)更多知識,歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權(quán)請聯(lián)系站長郵箱:is@yisu.com進(jìn)行舉報,并提供相關(guān)證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權(quán)內(nèi)容。