深入瞭解Java數據結構和算法之堆

1、堆的定義

①、它是完全二叉樹,除瞭樹的最後一層節點不需要是滿的,其它的每一層從左到右都是滿的。註意下面兩種情況,第二種最後一層從左到右中間有斷隔,那麼也是不完全二叉樹。

②、它通常用數組來實現。  

這種用數組實現的二叉樹,假設節點的索引值為index,那麼:

節點的左子節點是 2*index+1,

節點的右子節點是 2*index+2,

節點的父節點是 (index-1)/2。

③、堆中的每一個節點的關鍵字都大於(或等於)這個節點的子節點的關鍵字。

這裡要註意堆和前面說的二叉搜索樹的區別,二叉搜索樹中所有節點的左子節點關鍵字都小於右子節點關鍵字,在二叉搜索樹中通過一個簡單的算法就可以按序遍歷節點。但是在堆中,按序遍歷節點是很困難的,如上圖所示,堆隻有沿著從根節點到葉子節點的每一條路徑是降序排列的,指定節點的左邊節點或者右邊節點,以及上層節點或者下層節點由於不在同一條路徑上,他們的關鍵字可能比指定節點大或者小。所以相對於二叉搜索樹,堆是弱序的。

2、遍歷和查找

前面我們說瞭,堆是弱序的,所以想要遍歷堆是很困難的,基本上,堆是不支持遍歷的。

對於查找,由於堆的特性,在查找的過程中,沒有足夠的信息來決定選擇通過節點的兩個子節點中的哪一個來選擇走向下一層,所以也很難在堆中查找到某個關鍵字。

因此,堆這種組織似乎非常接近無序,不過,對於快速的移除最大(或最小)節點,也就是根節點,以及能快速插入新的節點,這兩個操作就足夠瞭。

3、移除

移除是指刪除關鍵字最大的節點(或最小),也就是根節點。

根節點在數組中的索引總是0,即maxNode = heapArray[0];

移除根節點之後,那樹就空瞭一個根節點,也就是數組有瞭一個空的數據單元,這個空單元我們必須填上。

第一種方法:將數組所有數據項都向前移動一個單元,這比較費時。

第二種方法:

  • ①、移走根
  • ②、把最後一個節點移動到根的位置
  • ③、一直向下篩選這個節點,直到它在一個大於它的節點之下,小於它的節點之上為止。

具體步驟如下:

圖a表示把最後一個節點移到根節點,圖b、c、d表示將節點向下篩選到合適的位置,它的合適位置在最底層(有時候可能在中間),圖e表示節點在正確位置的情景。

註意:向下篩選的時候,將目標節點和其子節點比較,誰大就和誰交換位置。

4、插入

插入節點也很容易,插入時,選擇向上篩選,節點初始時插入到數組最後第一個空著的單元,數組容量大小增一。然後進行向上篩選的算法。

註意:向上篩選和向下不同,向上篩選隻用和一個父節點進行比較,比父節點小就停止篩選瞭。

5、完整的Java堆代碼

首先我們要知道用數組表示堆的一些要點。若數組中節點的索引為x,則:

節點的左子節點是 2*index+1,

節點的右子節點是 2*index+2,

節點的父節點是 (index-1)/2。

註意:"/" 這個符號,應用於整數的算式時,它執行整除,且得到是是向下取整的值。

package com.ys.tree.heap;
 
public class Heap {
     
    private Node[] heapArray;
    private int maxSize;
    private int currentSize;
     
    public Heap(int mx) {
        maxSize = mx;
        currentSize = 0;
        heapArray = new Node[maxSize];
    }
     
    public boolean isEmpty() {
        return (currentSize == 0)? true : false;
    }
     
    public boolean isFull() {
        return (currentSize == maxSize)? true : false;
    }
     
    public boolean insert(int key) {
        if(isFull()) {
            return false;
        }
        Node newNode = new Node(key);
        heapArray[currentSize] = newNode;
        trickleUp(currentSize++);
        return true;
    }
    //向上調整
    public void trickleUp(int index) {
        int parent = (index - 1) / 2; //父節點的索引
        Node bottom = heapArray[index]; //將新加的尾節點存在bottom中
        while(index > 0 && heapArray[parent].getKey() < bottom.getKey()) {
            heapArray[index] = heapArray[parent];
            index = parent;
            parent = (parent - 1) / 2;
        }
        heapArray[index] = bottom;
    }
     
    public Node remove() {
        Node root = heapArray[0];
        heapArray[0] = heapArray[--currentSize];
        trickleDown(0);
        return root;
    }
    //向下調整
    public void trickleDown(int index) {
        Node top = heapArray[index];
        int largeChildIndex;
        while(index < currentSize/2) { //while node has at least one child
            int leftChildIndex = 2 * index + 1;
            int rightChildIndex = leftChildIndex + 1;
            //find larger child
            if(rightChildIndex < currentSize &&  //rightChild exists?
                    heapArray[leftChildIndex].getKey() < heapArray[rightChildIndex].getKey()) {
                largeChildIndex = rightChildIndex;
            }
            else {
                largeChildIndex = leftChildIndex;
            }
            if(top.getKey() >= heapArray[largeChildIndex].getKey()) {
                break;
            }
            heapArray[index] = heapArray[largeChildIndex];
            index = largeChildIndex;
        }
        heapArray[index] = top;
    }
    //根據索引改變堆中某個數據
    public boolean change(int index, int newValue) {
        if(index < 0 || index >= currentSize) {
            return false;
        }
        int oldValue = heapArray[index].getKey();
        heapArray[index].setKey(newValue);
        if(oldValue < newValue) {
            trickleUp(index);
        }
        else {
            trickleDown(index);
        }
        return true;
    }
     
    public void displayHeap() {
        System.out.println("heapArray(array format): ");
        for(int i = 0; i < currentSize; i++) {
            if(heapArray[i] != null) {
                System.out.print(heapArray[i].getKey() + " ");
            }
            else {
                System.out.print("--");
            }
        }
    }
}
class Node {
    private int iData;
    public Node(int key) {
        iData = key;
    }
     
    public int getKey() {
        return iData;
    }
     
    public void setKey(int key) {
        iData = key;
    }
}

總結

本篇文章就到這裡瞭,希望能夠給你帶來幫助,也希望您能夠多多關註WalkonNet的更多內容!

推薦閱讀: