二叉堆與優先隊列

一、優先隊列

1.簡單介紹

優先隊列是一種抽象的數據結構,它與我們生活中的許多場景息息相關。比如我們的電腦或者手機,很多時候我們后臺會運行多個程序,當程序過多導致內存急劇減少時,如果沒有相關權限的設置,處理器會優先關閉掛在后臺比較久的,然后繼續往后臺加入程序。和基本的數據結構隊列和棧類似,不過優先隊列對元素的操作是有優先級的,和上面描述的一樣,它的核心是刪除最大元素和插入元素。

2.基本實現

優先隊列的實現有很多方式,常見的初級實現有基于有序數組和無序數組的,以及基于鏈表的,都能達到優先隊列的核心操作。但這些實現平均效率不高。這里就重點描述下基于堆的優先隊列的實現。

二、二叉堆

1.簡單介紹

二叉堆是一組能夠用堆有序的完全二叉樹排序的元素,并在數組中按層級儲存(不使用數組的第一個元素)。且根結點是堆有序的二叉樹中最大的節點。在一個二叉堆中,如果一個節點的位置為k,則它的父結點的位置為k/2,左右子節點的位置分別為2k、2k+1。

2.基本實現

①比較和交換

這個比較簡單,這里就不多說了,直接上代碼

// 比較兩個元素大小
private boolean less(int i,int j) {
        return pq[i].compareTo(pq[j]) < 0;
    }
// 交換兩個元素
private void exch(int i,int j) {
        Key t = pq[i];
        pq[i] = pq[j];
        pq[j] = t;
    }

②插入和刪除最大元素

插入元素就直接將元素增加到堆的末尾,然后進行上浮操作,使二叉堆有序。

刪除最大元素就直接從二叉堆頂端刪除,然后進行下沉操作。

所以接下來2個將堆有序化的操作是比較重要的,因為不管是刪除最大元素還是插入元素后,都要繼續保持二叉堆的有序性。

③由下至上的堆的有序化

這個也稱上浮。如果二叉堆的有序化因為某個結點變得比它的父結點更加大而打破,就需要進行不斷的上浮操作,也就是把當前結點與父結點交換位置直到二叉堆有序為止。看看下面的代碼

private void swim(int k) {
        while (k > 1&&less(k/2,k)) {
            exch(k/2,k);
            k = k/2;
        }
    }

這里循環的條件判斷了只有當父結點比子結點更加小時交換才會繼續進行。

④由上至下的堆有序化

這個也稱下沉。與上浮類似,如果二叉堆的有序化因為某個結點變得比它的某個子結點更加小而打破,就需要進行不斷的下沉操作,把當前結點與其子結點中較大的部分進行交換(注意:二叉堆的左右結點是不需要按大小來排序的,但都比父結點小)。下面來看看代碼

private void sink(int k) {
        while (2*k <= N) {
            int j = 2*k;
            if (j < N&&less(j,j+1)) {
                j++;
            }
            if (!less(k,j)) break;
            exch(k,j);
            k = j;
        }
    }

這里注意一下第一個判斷,如果當前結點的左結點比右結點小,就進行++操作,這樣j就變成了右結點。繼續往下,如果當前結點比j結點大,就結束循環。否則就以j結點為當前結點繼續進行下一輪循環。

最后附上基于堆的優先隊列的全部代碼

/*
 * 基于堆的優先隊列
 */
public class MaxPQ<Key extends Comparable<Key>> {

    private Key[] pq;   //基于堆的完全二叉樹
    private int N = 0;  // 在數組中儲存的范圍是[1...N] 不包括pq[0]

    public MaxPQ(int MaxN) {
        pq = (Key[]) new Comparable[MaxN+1];
    }
    // 判斷數組是否為空
    public boolean isEmpty() {
        return N == 0;
    }
    // 數組大小
    public int size() {
        return N;
    }
    // 插入元素
    public void insert(Key v) {
        pq[++N] = v;
        swim(N);
    }
    // 刪除最大元素
    public Key delMax() {
        Key max = pq[1];
        exch(1,N--);
        pq[N+1] = null; //防止對象游離
        sink(1);
        return max;
    }
    // 上浮操作
    private void swim(int k) {
        while (k > 1&&less(k/2,k)) {
            exch(k/2,k);
            k = k/2;
        }
    }
    // 下沉操作
    private void sink(int k) {
        while (2*k <= N) {
            int j = 2*k;
            if (j < N&&less(j,j+1)) {
                j++;
            }
            if (!less(k,j)) break;
            exch(k,j);
            k = j;
        }
    }
    // 比較兩個元素大小
    private boolean less(int i,int j) {
        return pq[i].compareTo(pq[j]) < 0;
    }
    // 交換兩個元素
    private void exch(int i,int j) {
        Key t = pq[i];
        pq[i] = pq[j];
        pq[j] = t;
    }
}

然后看看優先隊列的各種實現的算法復雜度

數據結構 插入元素 刪除最大元素
有序數組 N 1
無序數組 1 N
logN logN

相信對排序算法了解的朋友,可以看出基于堆的優先隊列就等同于堆排序,這里就不多說了,最后說一句題外話,不管算法與數據結構有關的學習資料多么全,關鍵還是在于自己的多思考,多花時間,行動是理想最高貴的表達,梅花香自苦寒來,太急于求成只會得不償失,這句話也是給我自己的一個提醒。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 229,963評論 6 542
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,348評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,083評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,706評論 1 317
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,442評論 6 412
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,802評論 1 328
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,795評論 3 446
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,983評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,542評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,287評論 3 358
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,486評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,030評論 5 363
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,710評論 3 348
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,116評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,412評論 1 294
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,224評論 3 398
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,462評論 2 378

推薦閱讀更多精彩內容

  • 數據結構與算法--優先隊列和堆排序 在某些數據處理的例子中,總數據量太大,無法排序(甚至無法全部裝進內存)。例如,...
    sunhaiyu閱讀 1,048評論 0 2
  • 本文的目標是要做出優先隊列和堆排序兩個Demo。 完全二叉樹 優先隊列 堆排序 完全二叉樹 完全二叉樹的定義是建立...
    囧書閱讀 4,980評論 13 48
  • 0.目錄 1.優先隊列ADT 2.幾種實現 3.二叉堆 4.d-堆 5.左式堆 6.斜堆 7.二項隊列 8.斐波那...
    王偵閱讀 3,129評論 1 2
  • 優先隊列 支持刪除最大元素和插入元素兩種操作的數據結構可以稱之為優先隊列。可以用無序數組|有序數組(下壓棧), 鏈...
    melouverrr閱讀 773評論 0 0
  • 四. 走向世界之巔——快速排序 你可能會以為歸并排序是最強的算法了,其實不然。回想一下,歸并的時間效率雖然高,但空...
    Leesper閱讀 1,737評論 9 7