堆和堆排序

總結(jié)


再講堆和堆排序之前先講優(yōu)先隊列

優(yōu)先隊列

  • 優(yōu)先隊列是什么:
    與常見的隊列不同的是,優(yōu)先隊列并不遵循“先進先出”的原則,反而是根據(jù)優(yōu)先級來確定是否先出。優(yōu)先級高的先出,優(yōu)先級低的后出
  • 為什么使用優(yōu)先隊列:
    優(yōu)先隊列是用在,處理動態(tài)的數(shù)據(jù)排序。也就是優(yōu)先隊列所處理的數(shù)據(jù)可能時刻在變化的。

例子:醫(yī)院排隊,病情緊急的病人,先去治療。該情景中:1.病人是可能時刻在增加的,也就是動態(tài)的。其次,是按優(yōu)先級進行進出的

優(yōu)先隊列的實現(xiàn)

預(yù)先知識

為什么使用堆

和所有的隊列一樣,隊列包含兩種操作:1.入隊,2.出隊

使用普通數(shù)組或者順序數(shù)組也可以達到優(yōu)先隊列的操作,但是使用堆的效率最高

也許你的表情是

大佬息怒先看這個表(優(yōu)先級最高的數(shù)據(jù)進行操作的算法復(fù)雜度)

入隊 出隊
普通數(shù)組 O(1) O(n)
順序數(shù)組 O(n) O(1)
O(lgn) O(lgn)

從上表可以得到,用堆的算法復(fù)雜度是最低的。
在最差的情況下,使用數(shù)組是O(n^2),但是使用堆O(nlgn)
現(xiàn)在是不是應(yīng)該


堆的基礎(chǔ)知識

我們實現(xiàn)的方式是使用二叉堆(完全二叉樹)

完全二叉樹

特點:

  1. 每個根節(jié)點最多只有兩個子節(jié)點
  2. 子節(jié)點小于根節(jié)點
  3. 除去最后一層,其他層的節(jié)點必須為最大值
  4. 最后一層可以不是最大值,但它必須在二叉樹的最左邊

使用數(shù)組來完成一個堆

數(shù)組堆

由圖可以的到如下性質(zhì):

  1. 左子節(jié)點的索引是根節(jié)點的兩倍
  2. 右子節(jié)點的索引是根節(jié)點的兩倍加一

因此我們可以得到的公式:

  1. 得到當(dāng)前索引i的父節(jié)點i/2
  2. 得到當(dāng)前索引的左子節(jié)點i*2
  3. 得到當(dāng)前索引的右子節(jié)點i*2+1
  4. 得到當(dāng)前堆最后一個帶有子節(jié)點的父節(jié)點索引是conut/2其中conut為堆的長度

常用操作

插入一個新元素在堆的最后并排序shiftUp操作

shiftUp

如圖我們在最大堆中插入一個新元素30,那么它先與其父節(jié)點(父節(jié)點位置公式在上面)進行比較,也就是圖中的16,30明顯大于16,進行交換位置。新元素繼續(xù)和其父節(jié)點也就是41,進行比較大小。很顯然,小于41,那么shiftUp操作結(jié)束
在父節(jié)點中存在一個小于子節(jié)點的數(shù),要進行的操作shiftDown

shiftDown

如圖,一共為兩個步驟

  1. 左右子節(jié)點相互比較,選出最大值
  2. 與父節(jié)點進行比較,如果大于父節(jié)點進行替換

優(yōu)先隊列的代碼實現(xiàn)(使用堆的方式)

  • 滿足一個數(shù)據(jù)結(jié)構(gòu)
public class MaxHeap<Item extends Comparable> {
    protected Item[] datas;
    protected int count;       //元素的個數(shù)
   private int capacity;       //預(yù)先申請空間大小
 public MaxHeap(int capacity) {
        this.capacity = capacity;
  datas= (Item[]) new Comparable[capacity+1];//加一的原因是
  count=0;
  }
 public int size(){
        return count;
  }
    public boolean isEmpty(){
        return count == 0;
  }
}
  • 常用操作shiftUpshiftDown實現(xiàn)(具體操作描述在上面已經(jīng)講述)
private void shiftUp(int k) {
    while (k>1 && datas[k/2].compareTo(datas[k])<0){  //與它的父節(jié)點進行比較
        swap(k/2,k);     //交換位置
    k/=2;
  }
}
private void shiftDown(int k){
    while (2*k<=count){
        int j=2*k;
        if (j+1<=count && datas[j+1].compareTo(datas[j])>0)//兩個子節(jié)點比較
            j++;
        if (datas[k].compareTo(datas[j])>=0)//父節(jié)點與較大子節(jié)點比較
            break;
       swap(k,j);
        k=j;
  }
}
  • 插入一個元素(相當(dāng)于增加一個索引并執(zhí)行shiftUp操作)
public void insert(Item item){
     assert count + 1<=capacity;
     datas[count+1]=item;
     count++;
     shiftUp(count);
}
  • 出隊最大的元素(相當(dāng)于將最大元素與第conutconut為堆的長度)個元素進行交換,然后,進行shiftDown操作)
public Item extractMax(){
    assert count>0;
    Item ret=datas[1];
     swap(1,count);
     count--;
     shiftDown(1);
     return ret;
}

所有代碼

public class MaxHeap<Item extends Comparable> {
    protected Item[] datas;
    protected int count;
    private int capacity;
  public MaxHeap(int capacity) {
        this.capacity = capacity;
        datas= (Item[]) new Comparable[capacity+1];
        count=0;
  }
    public MaxHeap(Item[] arr,int n){
        datas= (Item[]) new Comparable[n];
        capacity=n;
        for (int i = 0; i < arr.length ; i++)
            datas[i+1]=arr[i];
            count=n;
       for (int i = count/2; i >=1; i--)
            shiftDown(i);

     }
    public int size(){
        return count;
  }
    public boolean isEmpty(){
        return count == 0;
  }
    public void insert(Item item){
        assert count + 1<=capacity;
        datas[count+1]=item;
        count++;
        shiftUp(count);
  }
    public Item extractMax(){
        assert count>0;
        Item ret=datas[1];
        swap(1,count);
        count--;
        shiftDown(1);
        return ret;
  }

    private void shiftUp(int k) {
        while (k>1 && datas[k/2].compareTo(datas[k])<0){
            swap(k/2,k);
        k/=2;
  }
    }
    private void shiftDown(int k){
        while (2*k<=count){
            int j=2*k;
        if (j+1<=count && datas[j+1].compareTo(datas[j])>0)
                j++;
        if (datas[k].compareTo(datas[j])>=0)
                break;
          swap(k,j);
          k=j;
       }
    }

    private void swap(int i, int count) {
        Item t=datas[i];
        datas[i]=datas[count];
        datas[count]=t;
  }
}

優(yōu)先隊列的優(yōu)化

起因:上例實現(xiàn)的優(yōu)先隊列,顯然需要創(chuàng)建新的空間。
解決方法:(總體使用數(shù)組的方式來解決)
  • 當(dāng)前堆的最大值與數(shù)組最后一個值進行交換


  • 交換后,除最后一個元素其他的元素進行ShiftDown操作,保持前面部分依舊是一個最大堆

  • 再重復(fù)所有步驟第一步操作


全部代碼(ShiftDown函數(shù)與上次的例子不同的原因是因為索引值是從0開始,而不是從1開始)

public static void LocalMaxHeapFunc(Comparable[] arr){
    //1.進行一次Heapify的過程
  int n=arr.length;
  //注意:其中我們的堆是從零開始索引的
  //從(最后一個元素的索引-1)/2開始
  //其中最后一個索引是n-1
  for (int i = (n-1-1)/2; i >=0 ; i--) {
        shiftDown(arr,n,i);
  }
    for (int i = n-1; i > 0; i--) {
        Utils.swap2(arr,0,i);
  shiftDown(arr,i,0);
  }
}

private static void shiftDown(Comparable[] datas,int count,int k) {
    while ((2*k+1)<count){
        int j=2*k+1;
 if (j+1< count && datas[j+1].compareTo(datas[j])>0)
            j++;
 if (datas[k].compareTo(datas[j])>=0)
            break;
  Utils.swap2(datas,k,j);
  k=j;
  }
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點,簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,345評論 6 531
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 98,494評論 3 416
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 176,283評論 0 374
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經(jīng)常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,953評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 71,714評論 6 410
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 55,186評論 1 324
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,255評論 3 441
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 42,410評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 48,940評論 1 335
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 40,776評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,976評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,518評論 5 359
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點故事閱讀 44,210評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,642評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,878評論 1 286
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,654評論 3 391
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 47,958評論 2 373

推薦閱讀更多精彩內(nèi)容

  • 一、基本知識 初始化堆:按照完全二叉樹的結(jié)構(gòu),從上到下、從左到右依次放入元素,然后可以認(rèn)為所有的葉子結(jié)點已經(jīng)符合大...
    鬼谷神奇閱讀 325評論 0 0
  • 堆的簡介 堆排序是一種復(fù)雜度為Nlog(N)的排序算法。介紹堆排序之前先講一講什么是堆。這里介紹的是數(shù)據(jù)結(jié)構(gòu)中的二...
    渭城一場雨閱讀 313評論 0 2
  • 堆: 堆是具有下列性質(zhì)的完全二叉樹:每個節(jié)點的值都大于或等于左右孩子節(jié)點的值,稱為大頂堆;每個節(jié)點的值都小于或等于...
    狗尾巴草敗了閱讀 148評論 0 0
  • 四. 走向世界之巔——快速排序 你可能會以為歸并排序是最強的算法了,其實不然。回想一下,歸并的時間效率雖然高,但空...
    Leesper閱讀 1,728評論 9 7
  • 這一刻我是眩暈的 因為我變了 變得不那么純粹和執(zhí)著 我沒有支撐起未來的能力 我希望能通過一點不滅的光輝掙取 但是我...
    正山小種ww閱讀 162評論 0 0