紙上談兵: 堆 (heap)

紙上談兵: 堆 (heap)

作者:Vamei 出處:http://www.cnblogs.com/vamei 歡迎轉載,也請保留這段聲明。謝謝!

?

堆(heap)又被為優先隊列(priority queue)。盡管名為優先隊列,但堆并不是隊列。回憶一下,在隊列中,我們可以進行的限定操作是dequeue和enqueue。dequeue是按照進入隊列的先后順序來取出元素。而在堆中,我們不是按照元素進入隊列的先后順序取出元素的,而是按照元素的優先級取出元素。

這就好像候機的時候,無論誰先到達候機廳,總是頭等艙的乘客先登機,然后是商務艙的乘客,最后是經濟艙的乘客。每個乘客都有頭等艙、商務艙、經濟艙三種個鍵值(key)中的一個。頭等艙->商務艙->經濟艙依次享有從高到低的優先級。

再比如,封建社會的等級制度,也是一個堆。在這個堆中,國王、貴族、騎士和農民是從高到低的優先級。

封建等級

?

Linux內核中的調度器(scheduler)會按照各個進程的優先級來安排CPU執行哪一個進程。計算機中通常有多個進程,每個進程有不同的優先級(該優先級的計算會綜合多個因素,比如進程所需要耗費的時間,進程已經等待的時間,用戶的優先級,用戶設定的進程優先程度等等)。內核會找到優先級最高的進程,并執行。如果有優先級更高的進程被提交,那么調度器會轉而安排該進程運行。優先級比較低的進程則會等待。“堆”是實現調度器的理想數據結構。

(Linux中可以使用nice命令來影響進程的優先級)

?

堆的實現

堆的一個經典的實現是完全二叉樹(complete binary tree)。這樣實現的堆成為二叉堆(binary heap)。

完全二叉樹是增加了限定條件的二叉樹。假設一個二叉樹的深度為n。為了滿足完全二叉樹的要求,該二叉樹的前n-1層必須填滿,第n層也必須按照從左到右的順序被填滿,比如下圖:

為了實現堆的操作,我們額外增加一個要求:?任意節點的優先級不小于它的子節點。如果在上圖中,設定小的元素值享有高的優先級,那么上圖就符合該要求。

這類似于“疊羅漢”。疊羅漢最重要的一點,就是讓體重大的參與者站在最下面,讓體重小的參與者站在上面 (體重小,優先級高)。為了讓“堆”穩固,我們每次只允許最上面的參與者退出堆。也就是,每次取出的優先級最高的元素。

三個“疊羅漢”堆

?

我已經在排序算法簡介及其C實現中實際使用了堆。堆的主要操作是插入和刪除最小元素(元素值本身為優先級鍵值,小元素享有高優先級)。在插入或者刪除操作之后,我們必須保持該實現應有的性質: 1. 完全二叉樹 2. 每個節點值都小于或等于它的子節點。

?

在插入操作的時候,會破壞上述堆的性質,所以需要進行名為percolate_up的操作,以進行恢復。新插入的節點new放在完全二叉樹最后的位置,再和父節點比較。如果new節點比父節點小,那么交換兩者。交換之后,繼續和新的父節點比較…… 直到new節點不比父節點小,或者new節點成為根節點。這樣得到的樹,就恢復了堆的性質。

我們插入節點2:

插入

?

刪除操作只能刪除根節點。根節點刪除后,我們會有兩個子樹,我們需要基于它們重構堆。進行percolate_down的操作: 讓最后一個節點last成為新的節點,從而構成一個新的二叉樹。再將last節點不斷的和子節點比較。如果last節點比兩個子節點中小的那一個大,則和該子節點交換。直到last節點不大于任一子節點都小,或者last節點成為葉節點。

刪除根節點1。如圖:

刪除根節點

?

下面是代碼。與我們在二叉搜索樹中使用表不同,我們這里使用數組來表示完全二叉樹。數組下標為0的元素不用于儲存節點,而用于記錄完全二叉樹中元素的總數。

復制代碼
/* By Vamei Use an big array to implement heapDECLARE: int heap[MAXSIZE] in calling functionheap[0] : total nodes in the heapfor a node i, its children are i*2 and i*2+1 (if exists)its parent is i/2  */void insert(int new, int heap[]) 
{int childIdx, parentIdx;heap[0] = heap[0] + 1;heap[heap[0]] = new;/* recover heap property */percolate_up(heap);
}static void percolate_up(int heap[]) {int lightIdx, parentIdx;lightIdx  = heap[0];parentIdx = lightIdx/2;/* lightIdx is root? && swap? */while((parentIdx > 0) && (heap[lightIdx] < heap[parentIdx])) {/* swap */swap(heap + lightIdx, heap + parentIdx); lightIdx  = parentIdx;parentIdx = lightIdx/2;}
}int delete_min(int heap[]) 
{int min;if (heap[0] < 1) {/* delete element from an empty heap */printf("Error: delete_min from an empty heap.");exit(1);}/* delete root move the last leaf to the root */min = heap[1];swap(heap + 1, heap + heap[0]);heap[0] -= 1;/* recover heap property */percolate_down(heap);return min;
}static void percolate_down(int heap[]) {int heavyIdx;int childIdx1, childIdx2, minIdx;int sign; /* state variable, 1: swap; 0: no swap */heavyIdx = 1;do {sign     = 0;childIdx1 = heavyIdx*2;childIdx2 = childIdx1 + 1;if (childIdx1 > heap[0]) {/* both children are null */break; }else if (childIdx2 > heap[0]) {/* right children is null */minIdx = childIdx1;}else {minIdx = (heap[childIdx1] < heap[childIdx2]) ?childIdx1 : childIdx2;}if (heap[heavyIdx] > heap[minIdx]) {/* swap with child */swap(heap + heavyIdx, heap + minIdx);heavyIdx = minIdx;sign = 1;}} while(sign == 1);
}
復制代碼

你可以嘗試一下構建自己的main函數,測試相關的操作。

?

總結

堆,優先級

插入元素,刪除最大優先級元素

?

歡迎繼續閱讀“紙上談兵: 算法與數據結構”系列。

轉載于:https://www.cnblogs.com/Alandre/p/3705483.html

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

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

相關文章

C語言 取模運算 保證正數,c語言的取模運算

我們對C的%運算知多少呢&#xff1f;當是正整數時&#xff0c;可能大家都知道。例如&#xff1a;5%3等于2, 3%5等于3。當存在負數時呢&#xff1f;先看看例子&#xff1a;例一&#xff1a;int main(){int x;x -6%5; printf("%2d/n",x);x 6%-5; printf("%2d/n&…

操作系統上機題目(多線程2)

主線程創建4個子線程T1、T2、T3、T4&#xff0c;主線程在4個子線程退出后&#xff0c;才退出線程T1、T2、T3、T4的運行時代碼如下&#xff1a; #include <unistd.h> // sleep函數聲明在該頭文件中void *T1_entry(void *arg) {sleep(2); // 睡眠2秒&#xff0c;不準刪除…

圖形GUI名稱

15句柄圖形(Handle Graphics)15.1圖形窗的產生和控制(Figure window creation and control)clf 清除當前圖close 關閉圖形figure 打開或創建圖形窗口gcf 獲得當前圖的柄openfig 打開圖形refresh 刷新圖形shg 顯示圖形窗15.2軸的產生和控制(Axis creation and control)axes 在任…

c語言編程非線性方程求解,c語言計算機編程三種方法求解非線性方程.doc

c語言計算機編程三種方法求解非線性方程.doc本 科 專 業 學 年 論 文題 目非線性方程求解比較姓 名 何 娟 專 業 計算機科學技術系 班 級 08 級本科(2)班 指 導 老 師 劉 曉 娜 完成日期 2010 年 11 月 21 日計算機學年專業論文 非線性方程求解- 1 -題 目非線性方程求解比較摘 …

最近用到這個強大的工具 PhysicsEditor (轉)

今天收到PhysicsEditor作者發過來的license key&#xff0c;所以順便把PhysicsEditor也嘗試了一下。主要是嘗試將PhysicsEditor與cocos2dx&#xff0c;box2d結合開發的一些步驟。之前大概網絡檢索了一下&#xff0c;知道PhysicsEditor的功能其實很簡單。一句話就是給圖片的邊緣…

淺談塊級元素和內聯元素的嵌套規則

1. 替換和不可替換元素 從元素本身的特點來講&#xff0c;可以分為替換和不可替換元素。 a) 替換元素 替換元素就是瀏覽器根據元素的標簽和屬性&#xff0c;來決定元素的具體顯示內容。 例如瀏覽器會根據<img>標簽的src屬性的值來讀取圖片信息并顯示出來&#xff0c;而如…

如何更新Chrome

在瀏覽器的地址欄中輸入chrome://help即可進行自動更新&#xff0c;如下圖&#xff1a;

c語言用指針實現打開和關閉文件,我用rewind函數沒把指針直到開始,關閉文件然后打開就行。幫忙看看...

該樓層疑似違規已被系統折疊 隱藏此樓查看此樓高手啊&#xff0c;我試了&#xff0c;結果是回到文件頭了&#xff0c;把123覆蓋了&#xff0c;為ABC 但我這個程序就能直接不關閉文件而用rewind函數到開頭&#xff0c;你幫忙看看&#xff0c; //二進制文件的輸入輸出--數據塊的讀…

最佳適應算法模擬內存分配

最佳適應算法 從全部空閑區中找出能滿足作業要求的&#xff0c;且大小最小的空閑分區&#xff0c;這種方法能使碎片盡量小。 問題描述 Given five memory partitions of 100 KB, 500 KB, 200 KB, 300 KB, and 600 KB (in order), how would each of the first-fit, best-fit…

單片機c語言 i%3c%3c1,單片機C語言作業及上機習題及答案

《單片機C語言作業及上機習題及答案》由會員分享&#xff0c;可在線閱讀&#xff0c;更多相關《單片機C語言作業及上機習題及答案(37頁珍藏版)》請在人人文庫網上搜索。1、第一次課熟悉winTC編譯環境、熟悉C語言程序結構1.使用C 語言編譯環境&#xff0c;輸入下面的源程序。將你…

基于順序搜索的動態分區分配算法模擬內存動態分配--最佳適應算法(best fit,BF)

BF算法、男朋友算法&#xff0c;哈哈 要實現動態分區分配&#xff0c;需要考慮三個方面的問題。分別是數據結構、分區分配算法、分區的分配與回收操作。 首數據結構 這里我們使用的是空閑分區鏈&#xff0c;采用雙向鏈表表示空閑分區。 具體實現如下&#xff1a; typedef …

我也要談談大型網站架構之系列(4)——分布式中的異步通信

我們知道在面向對象編程中&#xff0c;總會想著各種辦法來實現代碼的解耦&#xff0c;從而讓項目中的各種人員面對自己熟悉的業務進行開發&#xff0c; 做到術業有專攻&#xff0c;比如大家非常熟悉的三層架構&#xff0c;MVC&#xff0c;MVP以及MVVM模式&#xff0c;讓前端設計…

node模塊函數圖解

已截圖方式記錄模塊信息&#xff1a; HTTP模塊&#xff1a; 對于網絡返回處理狀態封裝了很多種&#xff0c;我已截圖展現 以上狀態也是在http協議中包含的狀態。 http函數&#xff1a; path模塊&#xff1a; 轉載于:https://www.cnblogs.com/kuailingmin/p/4547538.html

android 心跳效果動畫,Android實現心跳的效果

最近再做一個教育類的項目。在做一些學習工具的時候&#xff0c;美工提出了一些要求&#xff0c;大致如下&#xff1a;其實實現過程也不難&#xff0c;大致就是對一個視圖控件添加一個圓形的背景&#xff0c;然后該視圖進行動畫處理&#xff0c;膨脹的同時&#xff0c;透明度增…

Oracle超出最大連接數問題及解決

用過Oracle的應該都熟悉如何查看和設置Oracle數據庫的最大連接數。這里就再啰嗦一遍。 查看當前的連接數&#xff0c;可以用select count(*) from v$process;設置的最大連接數&#xff08;默認值為150&#xff09;select value from v$parameter where name ‘processes’;修改…

操作系統上機作業--使用系統調用實現mycat

mycat.c的功能與系統cat程序相同mycat將指定的文件內容輸出到屏幕&#xff0c;例子如下&#xff1a;要求使用系統調用open/read/write/close實現 $ cat /etc/passwd root:x:0:0:root:/root:/bin/bash daemon:x:1:1:daemon:/usr/sbin:/usr/sbin/nologin bin:x:2:2:bin:/bin:/u…

GCDAynscSocket簡單使用-客戶端

這是一篇介紹GCDAynscSocket客戶端簡單使用的文章&#xff08;服務端后續添加&#xff09; 背景&#xff1a;在這篇文章之前我對socket的了解僅限于知道有TCP、UDP兩種方式&#xff0c;使用抓包工具時甚至看不懂抓包數據&#xff08;慚愧...&#xff09;&#xff0c;所以本文介…

微信android版字體,微信炫彩字下載-微信七彩字體 安卓版v1.6.2-PC6安卓網

微信七彩字體一款方便的手機字體更換軟件&#xff0c;微信炫彩字軟件集合了上百款優質中文美化字體&#xff0c;微信七彩發光字里有可愛的喵嗚體、卡通體&#xff0c;清秀的靜蕾體等多種字體。軟件介紹微信、qq上最好用、最個性的聊天字體應用&#xff0c;讓你的聊天與眾不同&a…

Android SQLite 數據庫 增刪改查操作

Android SQLite 數據庫 增刪改查操作 轉載▼一、使用嵌入式關系型SQLite數據庫存儲數據在Android平臺上&#xff0c;集成了一個嵌入式關系型數據庫——SQLite&#xff0c;SQLite3支持NULL、INTEGER、REAL&#xff08;浮點數字&#xff09;、TEXT(字符串文本)和BLOB(二進制對象…

SIT與UAT的分別

在企業級軟件的測試過程中&#xff0c;經常會劃分為三個階段——單元測試&#xff0c;SIT和UAT&#xff0c;如果開發人員足夠&#xff0c;通常還會在SIT之前引入代碼審查機制&#xff08;Code Review&#xff09;來保證軟件符合客戶需求且流程正確。下面簡單介紹一下SIT和UAT的…