15 張圖剖析內(nèi)存分配之malloc詳解
前言
由于malloc()的源碼十分的繁瑣,并且會(huì)調(diào)用OS所提供的API,所以我不在對(duì)malloc()的源碼進(jìn)行分析了,而只是會(huì)分析malloc()的動(dòng)作,這就已經(jīng)足夠了。
一、malloc()分配出的內(nèi)存空間
在前邊的文章中已經(jīng)提及到了,當(dāng)malloc()分配空間時(shí),并不是要多少就分配多少,而是會(huì)額外的加上首部和尾部,其中一些較為簡(jiǎn)單的部分我會(huì)在這里進(jìn)行解釋,而較為重要的部分我會(huì)在本文下面的分析中逐步的完善。圖片取自侯捷C 內(nèi)存分配系列教程講義
- 1、2兩塊空間保存了兩根指針,目的是使多個(gè)內(nèi)存塊連接成鏈表。
- 3空間保存了申請(qǐng)本內(nèi)存塊的文件名
- 4- 空間保存了申請(qǐng)本內(nèi)存塊的代碼行數(shù)
- 5空間記錄了本內(nèi)存塊中實(shí)際可以被用戶使用的內(nèi)存空間的大小
- 6空間記錄了當(dāng)前內(nèi)存塊的流水號(hào),即是鏈表中的第幾個(gè),從1開始
- 7空間記錄了當(dāng)前內(nèi)存塊被分配的形式,后邊會(huì)進(jìn)行分析
二、內(nèi)存分配
1.內(nèi)存管理所用到的結(jié)構(gòu)層次
首先,在進(jìn)入程序之前,系統(tǒng)就已經(jīng)分配出了一個(gè)結(jié)構(gòu)去管理內(nèi)存,我們先來(lái)看看這個(gè)結(jié)構(gòu)
- 指針pHeapData將被指向這個(gè)header所管理的那1MB的內(nèi)存空間的開頭。
- pRegion將會(huì)被指向一個(gè)管理用的結(jié)構(gòu),這個(gè)結(jié)構(gòu)將會(huì)在下邊展開
-
indGroupUse表示了當(dāng)前會(huì)提供內(nèi)存的group編號(hào),從0開始
-
cntRegionSize[64]用64個(gè)字節(jié)去對(duì)應(yīng)后邊group所將會(huì)展開鏈表,當(dāng)對(duì)應(yīng)鏈表掛在有內(nèi)存時(shí),將會(huì)變成1.
-
bitvGroupHi和bitvGroupLo共同構(gòu)成了一個(gè)的byteMap共64個(gè)byte(分為32組),將來(lái)用于對(duì)應(yīng)每個(gè)group中所掛載的64條雙向鏈表,當(dāng)對(duì)應(yīng)的位置掛載有內(nèi)存時(shí),會(huì)變成1.
-
grpHeadList就是32個(gè)group,每個(gè)group負(fù)責(zé)32KB
-
這里的cntEntries代表當(dāng)前鏈表中掛載的內(nèi)存塊被切分的次數(shù)
-
listHead對(duì)應(yīng)64對(duì)指針,也就是形成了64條鏈表,用于掛載不同大小的內(nèi)存塊,間隔為16byte,最后一條鏈表將掛載所有大于等于1K的內(nèi)存塊
-
編號(hào)1是當(dāng)前header所管理的1MB的空間,將其32等分,每一份的32KB由一個(gè)group去負(fù)責(zé)分配
-
編號(hào)2是一個(gè)group所管理的32K的空間,將其分為8個(gè)4KB大小的內(nèi)存頁(yè)掛載于最后一條鏈表上
-
編號(hào)3是分割好的內(nèi)存頁(yè)鏈表,他們被串成一個(gè)雙向鏈表。
-
編號(hào)4是一個(gè)group中的64條鏈表
2.內(nèi)存頁(yè)的劃分
下面我們來(lái)看每個(gè)嶄新的內(nèi)存頁(yè)的內(nèi)容
- 中間的空白區(qū)域代表了可共malloc()索取的4080byte的內(nèi)存空間
- 空白的最下邊和紅色的最上邊,兩個(gè)標(biāo)有4080的空間是用來(lái)記錄剩余可用空間大小的cookie
- 剩余的兩塊紅色部分是兩根指針,指向鏈表中前邊和后邊的內(nèi)存頁(yè)
- 黃色的標(biāo)有0xfdfdfdfd的是兩根分割區(qū)域,具體作用上邊已經(jīng)提及
- 最上邊的保留區(qū)域是為了讓下邊空白區(qū)域成為16byte的整數(shù)倍
內(nèi)存頁(yè)劃分的規(guī)則
當(dāng)申請(qǐng)一個(gè)內(nèi)存空間時(shí),首先先去符合的鏈表中尋找,如果鏈表中沒有掛載內(nèi)存塊,就從編號(hào)較大的鏈表中最近的掛有內(nèi)存塊的鏈表中劃分。
內(nèi)存頁(yè)被劃分之后的情況
最左邊原先是一個(gè)嶄新的內(nèi)存頁(yè)(4K = ff0),然后我們從內(nèi)存頁(yè)中劃分出0x130 byte的空間:
- 編號(hào)為1的是被劃分出的實(shí)際空間
- 編號(hào)2是實(shí)際可以為用戶所使用的實(shí)際空間,這個(gè)空間應(yīng)該是0x100
- 上下兩根cookie記錄了被劃分出去的實(shí)際空間,至于為什么是0x131,之前的文章有提及
- 內(nèi)存被劃分出去后,malloc()再對(duì)其進(jìn)行復(fù)寫,然后將實(shí)際空間交付給客戶。
3.內(nèi)存分配的動(dòng)作
我們剛剛分配出了0x130的空間,我們先看看這個(gè)空間分配出去之后的動(dòng)作
- 編號(hào)1:此時(shí)由group0分配內(nèi)存,所以Region 中的 indGroupUse被設(shè)置為0
- 編號(hào)2:整個(gè)group的內(nèi)存頁(yè)被劃分了一次,所以Group 中的 cntEntries被置為1
- 編號(hào)3:此時(shí)group0只有最后一個(gè)鏈表空間上掛載了鏈表,所以Region 中對(duì)應(yīng)的byte被置為1
- 編號(hào)1:由于當(dāng)前是group1再分配內(nèi)存,所以Region 中的 indGroupUse設(shè)置為1
- 編號(hào)2:將group1中最后一條鏈表再bitMap中對(duì)應(yīng)的位設(shè)置為1
- 編號(hào)3:group1整個(gè)的內(nèi)存頁(yè)被劃分了一次,所以Group 中的 cntEntries被置為1
4.內(nèi)存歸還的動(dòng)作
當(dāng)多次連續(xù)分配之后,出現(xiàn)了一次歸還空間的動(dòng)作
- 編號(hào)1:當(dāng)前group分配出的內(nèi)存塊-1
- 編號(hào)2:由于此次歸還的內(nèi)存大小為0x240應(yīng)該掛載于第35號(hào)鏈表,所以將第35號(hào)鏈表對(duì)應(yīng)的bite設(shè)為1(這里將byteMap中每四個(gè)byte寫成了一個(gè)16進(jìn)制數(shù))
- 編號(hào)3:當(dāng)前還是group所分配內(nèi)存,所以所以Region 中的 indGroupUse仍為0
- 編號(hào)4:這時(shí)被歸還的內(nèi)存被復(fù)寫,兩個(gè)cookie從0x241變回0x240,表示沒有被使用,兩根指針連入35號(hào)鏈表。
三、將內(nèi)存歸還給OS
我們來(lái)探討幾個(gè)問題:
Q1、當(dāng)多個(gè)group被啟用時(shí),怎么去尋找歸還的內(nèi)存屬于哪個(gè)group?
答案很簡(jiǎn)單,夾殺法:我們知道每一個(gè)group對(duì)應(yīng)內(nèi)存的起始地址和結(jié)尾地址,我們只需要去判斷被歸還的指針中地址的大小是否在這二者之間,就能判斷出是否屬于當(dāng)前的group。而去尋找所對(duì)應(yīng)的header的方法也是如此。
Q2、怎么將內(nèi)存還給操作系統(tǒng)?`
這里時(shí)malloc和之前講過的分配器本質(zhì)上的區(qū)別,我們能將收回的內(nèi)存還給操作系統(tǒng),具體步驟如下:
- 對(duì)于回收的連續(xù)的內(nèi)存空間進(jìn)行合并 這個(gè)實(shí)現(xiàn)時(shí)基于上下兩個(gè)cookie的實(shí)現(xiàn)完成的
- 判斷分配的空間的全回收
- 當(dāng)內(nèi)存全回收之后的狀態(tài)
Q3:當(dāng)一個(gè)group全回收之后,我們需要將他立刻還給系統(tǒng)么?
答案肯定是否定的,因?yàn)槿绻覀內(nèi)厥找粋€(gè)就還一個(gè),那么當(dāng)下一次在需要分配時(shí),我們還需要重新分配。所以全回收的group不會(huì)立刻被還給系統(tǒng),而是等待下一個(gè)全回收的group出現(xiàn),就會(huì)將前一個(gè)group對(duì)應(yīng)的內(nèi)存free掉。
???????????????? END ????????????????