伙伴算法

通常情況下,一個高級操作系統必須要給進程提供基本的、能夠在任意時刻申請和釋放任意大小內存的功能,就像malloc 函數那樣,然而,實現malloc 函數并不簡單,由于進程申請內存的大小是任意的,如果操作系統對malloc 函數的實現方法不對,將直接導致一個不可避免的問題,那就是內存碎片。

內存碎片就是內存被分割成很小很小的一些塊,這些塊雖然是空閑的,但是卻小到無法使用。隨著申請和釋放次數的增加,內存將變得越來越不連續。最后,整個內存將只剩下碎片,即使有足夠的空閑頁框可以滿足請求,但要分配一個大塊的連續頁框就可能無法滿足,所以減少內存浪費的核心就是盡量避免產生內存碎片。

針對這樣的問題,有很多行之有效的解決方法,其中伙伴算法被證明是非常行之有效的一套內存管理方法,因此也被相當多的操作系統所采用。

伙伴算法,簡而言之,就是將內存分成若干塊,然后盡可能以最適合的方式滿足程序內存需求的一種內存管理算法,伙伴算法的一大優勢是它能夠完全避免外部碎片的產生。什么是外部碎片以及內部碎片,前面博文slab分配器后面已有介紹。申請時,伙伴算法會給程序分配一個較大的內存空間,即保證所有大塊內存都能得到滿足。很明顯分配比需求還大的內存空間,會產生內部碎片。所以伙伴算法雖然能夠完全避免外部碎片的產生,但這恰恰是以產生內部碎片為代價的。

Linux 便是采用這著名的伙伴系統算法來解決外部碎片的問題。把所有的空閑頁框分組為 11 塊鏈表,每一塊鏈表分別包含大小為1,2,4,8,16,32,64,128,256,512 和 1024 個連續的頁框。對1024 個頁框的最大請求對應著 4MB 大小的連續RAM 塊。每一塊的第一個頁框的物理地址是該塊大小的整數倍。例如,大小為 16個頁框的塊,其起始地址是 16 * 2^12 (2^12 = 4096,這是一個常規頁的大小)的倍數。

下面通過一個簡單的例子來說明該算法的工作原理:

假設要請求一個256(129~256)個頁框的塊。算法先在256個頁框的鏈表中檢查是否有一個空閑塊。如果沒有這樣的塊,算法會查找下一個更大的頁塊,也就是,在512個頁框的鏈表中找一個空閑塊。如果存在這樣的塊,內核就把512的頁框分成兩等分,一般用作滿足需求,另一半則插入到256個頁框的鏈表中。如果在512個頁框的塊鏈表中也沒找到空閑塊,就繼續找更大的塊——1024個頁框的塊。如果這樣的塊存在,內核就把1024個頁框塊的256個頁框用作請求,然后剩余的768個頁框中拿512個插入到512個頁框的鏈表中,再把最后的256個插入到256個頁框的鏈表中。如果1024個頁框的鏈表還是空的,算法就放棄并發出錯誤信號。

簡而言之,就是在分配內存時,首先從空閑的內存中搜索比申請的內存大的最小的內存塊。如果這樣的內存塊存在,則將這塊內存標記為“已用”,同時將該內存分配給應用程序。如果這樣的內存不存在,則操作系統將尋找更大塊的空閑內存,然后將這塊內存平分成兩部分,一部分返回給程序使用,另一部分作為空閑的內存塊等待下一次被分配。

以上過程的逆過程就是頁框塊的釋放過程,也是該算法名字的由來。內核試圖把大小為 b 的一對空閑伙伴塊合并為一個大小為 2b 的單獨塊。滿足以下條件的兩個塊稱為伙伴:

  • 兩個塊具有相同的大小,記作 b
  • 它們的物理地址是連續的
  • 第一塊的第一個頁框的物理地址是 2 * b * 2^12 的倍數

該算法是迭代的,如果它成功合并所釋放的塊,它會試圖合并 2b 的塊,以再次試圖形成更大的塊。

下面通過一個例子,來深入地理解一下伙伴算法的真正內涵(下面這個例子并不嚴格表示Linux 內核中的實現,是闡述伙伴算法的實現思想):

假設系統中有 1MB 大小的內存需要動態管理,按照伙伴算法的要求:需要將這1M大小的內存進行劃分。這里,我們將這1M的內存分為 64K、64K、128K、256K、和512K 共五個部分,如下圖 a 所示


1.此時,如果有一個程序A想要申請一塊45K大小的內存,則系統會將第一塊64K的內存塊分配給該程序(產生內部碎片為代價),如圖b所示;

2.然后程序B向系統申請一塊68K大小的內存,系統會將128K內存分配給該程序,如圖c所示;

3.接下來,程序C要申請一塊大小為35K的內存。系統將空閑的64K內存分配給該程序,如圖d所示;

4.之后程序D需要一塊大小為90K的內存。當程序提出申請時,系統本該分配給程序D一塊128K大小的內存,但此時內存中已經沒有空閑的128K內存塊了,于是根據伙伴算法的原理,系統會將256K大小的內存塊平分,將其中一塊分配給程序D,另一塊作為空閑內存塊保留,等待以后使用,如圖e所示;

5.緊接著,程序C釋放了它申請的64K內存。在內存釋放的同時,系統還負責檢查與之相鄰并且同樣大小的內存是否也空閑,由于此時程序A并沒有釋放它的內存,所以系統只會將程序C的64K內存回收,如圖f所示;

6.然后程序A也釋放掉由它申請的64K內存,系統隨機發現與之相鄰且大小相同的一段內存塊恰好也處于空閑狀態。于是,將兩者合并成128K內存,如圖g所示;

7.之后程序B釋放掉它的128k,系統也將這塊內存與相鄰的128K內存合并成256K的空閑內存,如圖h所示;

8.最后程序D也釋放掉它的內存,經過三次合并后,系統得到了一塊1024K的完整內存,如圖i所示。

https://blog.csdn.net/wenqian1991/article/details/27968779

本文來自互聯網用戶投稿,該文觀點僅代表作者本人,不代表本站立場。本站僅提供信息存儲空間服務,不擁有所有權,不承擔相關法律責任。
如若轉載,請注明出處:http://www.pswp.cn/news/382630.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/382630.shtml
英文地址,請注明出處:http://en.pswp.cn/news/382630.shtml

如若內容造成侵權/違法違規/事實不符,請聯系多彩編程網進行投訴反饋email:809451989@qq.com,一經查實,立即刪除!

相關文章

C++ 菱形虛繼承 通過指針來尋找繼承過來的成員變量

#define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std;//動物類 class Animal { public:int m_Age; //年齡 };//virtual加上后 繼承方式 數據虛繼承 // Animal類 變為 虛基類 //羊類 class Sheep : virtual public Animal {};//駝 class Tuo : virt…

CRC冗余校驗舉例和原理

什么是CRC校驗&#xff1f;CRC即循環冗余校驗碼&#xff1a;是數據通信領域中最常用的一種查錯校驗碼&#xff0c;其特征是信息字段和校驗字段的長度可以任意選定。循環冗余檢查&#xff08;CRC&#xff09;是一種數據傳輸檢錯功能&#xff0c;對數據進行多項式計算&#xff0c…

100篇打點!

原創終于到100了&#xff0c;寫一篇博客打點。在記錄一個很嚴重的問題&#xff0c;昨天面試&#xff0c;程序的思路都有了&#xff0c;可是在線OJ半天無法將多個字符串輸入并保存&#xff0c;遍歷。現在記錄一下方法&#xff01; #include <stdio.h> #include <stdli…

排序算法1快速排序

文章沒有解釋和代碼注釋&#xff0c;代碼經改進&#xff0c;做成了好理解,關鍵是好記憶的方式進行書寫。用于自己進行查閱 #include <stdio.h>void sort(int arr[] ,int left ,int right) {if(left > right)return;int i left;int j right;int get arr[right];whi…

C++ 多態原理初步01

當父類 Animal 的speak 前面加上 virtual 關鍵字之后&#xff0c;這個speak函數就變成了虛函數&#xff0c;Animal類結構發生了變化&#xff0c; 有了一個vfptr &#xff08;虛函數指針&#xff09;&#xff0c;指向了vftable&#xff08;虛函數表&#xff09;, 這個虛函數表里…

排序算法2歸并排序

文章沒有解釋和代碼注釋&#xff0c;代碼經改進&#xff0c;做成了好理解,關鍵是好記憶的方式進行書寫。用于自己進行查閱 #include <stdio.h>void merge(int arr1[],int left ,int mid ,int right) {int temp[sizeof(arr1)];int i left ;int j mid 1;int t 0;while…

C++ 多態之純虛函數和抽象類01

純虛函數的語法&#xff0c; virtual void func() 0;如果類中有了純虛函數&#xff0c; 那么這個類也成為抽象類抽象類無法實例化對象繼承了抽象類的子類&#xff0c;必須要重寫父類中的純虛函數&#xff0c;否則的話&#xff0c;子類也是屬于抽象類&#xff0c;無法實例化

堆排序面試

#文章沒有解釋和代碼注釋&#xff0c;代碼經改進&#xff0c;做成了好理解,關鍵是好記憶的方式進行書寫。用于自己進行查閱 #include <stdio.h>void swap(int arr[],int i,int j) {int temp arr[i];arr[i] arr[j];arr[j] temp; }void heapify(int arr[],int i,int si…

C++ 多態之虛析構與純虛擬購01

class Animal { public:Animal(){cout << "Animal的構造函數調用" << endl;}//虛析構 解決的問題是 當子類中有堆區內容&#xff0c;釋放時候對導致釋放不干凈&#xff0c;內存泄露//virtual ~Animal()//{// cout << "Animal的析構函數調用&…

面向對象與面向過程的本質的區別

https://blog.csdn.net/jerry11112/article/details/79027834 如果你很想搞明白面向對象是什么&#xff0c;面向過程是什么&#xff0c;或者說二者之間的區別是什么&#xff0c;那么就花費一點時間來研讀一下這篇博客&#xff0c;你一定會有很大的收獲的&#xff01; 一、面向…

C++ 向上轉型初步01

1.編譯器通過指針來訪問成員變量&#xff0c;指針指向哪個對象就使用哪個對象的數據&#xff1b;編譯器通過指針的類型來訪問成員函數&#xff0c;指針屬于哪個類的類型就使用哪個類的函數。 但是父類 函數如果變成虛函數&#xff0c;子類重寫了這個函數&#xff0c; 那么現象…

虛函數和純虛函數詳解

https://mp.weixin.qq.com/s?__bizMzAxNzYzMTU0Ng&mid2651289202&idx1&sn431ffd1fae4823366a50b68aed2838d4&chksm80114627b766cf31f72018ef5f1fe29591e9f6f4bd72018e7aea849342ca6f0a271fb38465ae#rd 打開鏈接看。轉載文章&#xff0c;注明出處 <p>學…

C++ 繼承中的同名成員的情況01

class Base { public:Base(){this->m_A 100;}void func(){cout << "Base中的Func調用" << endl;}void func(int a){cout << "Base中的Func(int a)調用" << endl;}int m_A; }; class Son : public Base { public:Son(){this-&g…

進程前臺運行后臺運行的相關命令

command& 讓進程在后臺運行jobs 查看后臺運行的進程fg %n 讓后臺運行的進程n到前臺來bg %n 讓進程n到后臺去&#xff1b; ctrl z 可以將一個正在前臺執行的命令放到后臺&#xff0c;并且暫停

西安電子科技大學求職打點

這兩天 一直在西安電子科技大學找工作&#xff0c;感覺自己漸漸失去了學習的狀態&#xff0c;本來很 多會的知識點都已經不會了。   今天休息一天&#xff0c;沒有去招聘會&#xff0c;看了看相關的知識&#xff0c;做了做題。也希望自己盡快恢復學習的感覺&#xff0c;溫故…

C++ 泛型編程模板 之 函數模板初步01

#define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std;void mySwapInt(int &a, int &b) {int temp a;a b;b temp; } void mySwapDouble(double &a, double &b) {double temp a;a b;b temp; } //利用模板實現通用交換函數 temp…

grep參數說明及常用用法

grep參數說明及常用用法 查看文件內容 [koulocalhost ~]$ more size.txt b124230 b034325 a081016 m7187998 m7282064 a022021 a061048 m9324822 b103303 a013386 b044525 m8987131 B081016 M45678 B103303 BADc2345 [] : 查看符合范圍內的信息 [koulocalho…

C++ 普通函數與函數模板 區別以及調用規則01

//普通函數 和 函數模板 區別 int myPlus(int a, int b) {return a b; }template<class T> T myPlus2(T a, T b) {return a b; }void test01() {int a 10;int b 20;char c c;cout << myPlus(a, c) << endl; //隱式類型轉換 將 char c轉為 int類型//myP…

C++ 模板的局限性以及解決01

#define _CRT_SECURE_NO_WARNINGS #include<iostream> using namespace std; #include <string>class Person { public:Person(string name, int age){this->m_Name name;this->m_Age age;}string m_Name;int m_Age; };//通過模板進行兩個數據比較 templat…

進程的狀態與種類

● 運行&#xff1a;正占用處理器   ● 就緒&#xff1a;只要獲得處理器即可運行。   ● 阻塞&#xff1a;正等待某個事件&#xff08;如I/O完成&#xff09;的發生。  在不少系統中&#xff0c;還增加了兩種基本狀態&#xff1a;   ● 新狀態&#xff1a;一個進程剛剛…