PriorityQueue

PriorityQueue

總體介紹

前面以Java ArrayDeque為例講解了StackQueue,其實還有一種特殊的隊列叫做PriorityQueue硕舆,即優(yōu)先隊列。優(yōu)先隊列的作用是能保證每次取出的元素都是隊列中權(quán)值最小的(Java的優(yōu)先隊列每次取最小元素,C++的優(yōu)先隊列每次取最大元素)杨名。這里牽涉到了大小關(guān)系熊经,元素大小的評判可以通過元素本身的自然順序(natural ordering)荠察,也可以通過構(gòu)造時傳入的比較器Comparator,類似于C++的仿函數(shù))奈搜。

Java中PriorityQueue實現(xiàn)了Queue接口悉盆,不允許放入null元素;其通過堆實現(xiàn)馋吗,具體說是通過完全二叉樹(complete binary tree)實現(xiàn)的小頂堆(任意一個非葉子節(jié)點的權(quán)值焕盟,都不大于其左右子節(jié)點的權(quán)值),也就意味著可以通過數(shù)組來作為PriorityQueue的底層實現(xiàn)宏粤。

上圖中我們給每個元素按照層序遍歷的方式進行了編號脚翘,如果你足夠細心,會發(fā)現(xiàn)父節(jié)點和子節(jié)點的編號是有聯(lián)系的绍哎,更確切的說父子節(jié)點的編號之間有如下關(guān)系:

leftNo = parentNo*2+1

rightNo = parentNo*2+2

parentNo = (nodeNo-1)/2

通過上述三個公式来农,可以輕易計算出某個節(jié)點的父節(jié)點以及子節(jié)點的下標。這也就是為什么可以直接用數(shù)組來存儲堆的原因崇堰。

PriorityQueuepeek()element操作是常數(shù)時間沃于,add(), offer(), 無參數(shù)的remove()以及poll()方法的時間復雜度都是log(N)

方法剖析

add()和offer()

add(E e)offer(E e)的語義相同海诲,都是向優(yōu)先隊列中插入元素繁莹,只是Queue接口規(guī)定二者對插入失敗時的處理不同,前者在插入失敗時拋出異常特幔,后則則會返回false咨演。對于PriorityQueue這兩個方法其實沒什么差別。

新加入的元素可能會破壞小頂堆的性質(zhì)蚯斯,因此需要進行必要的調(diào)整薄风。

//offer(E e)
public boolean offer(E e) {
    if (e == null)//不允許放入null元素
        throw new NullPointerException();
    modCount++;
    int i = size;
    if (i >= queue.length)
        grow(i + 1);//自動擴容
    size = i + 1;
    if (i == 0)//隊列原來為空,這是插入的第一個元素
        queue[0] = e;
    else
        siftUp(i, e);//調(diào)整
    return true;
}

上述代碼中拍嵌,擴容函數(shù)grow()類似于ArrayList里的grow()函數(shù)遭赂,就是再申請一個更大的數(shù)組,并將原數(shù)組的元素復制過去撰茎,這里不再贅述嵌牺。需要注意的是siftUp(int k, E x)方法,該方法用于插入元素x并維持堆的特性。

//siftUp()
private void siftUp(int k, E x) {
    while (k > 0) {
        int parent = (k - 1) >>> 1;//parentNo = (nodeNo-1)/2
        Object e = queue[parent];
        if (comparator.compare(x, (E) e) >= 0)//調(diào)用比較器的比較方法
            break;
        queue[k] = e;
        k = parent;
    }
    queue[k] = x;
}

新加入的元素x可能會破壞小頂堆的性質(zhì)逆粹,因此需要進行調(diào)整募疮。調(diào)整的過程為:從k指定的位置開始,將x逐層與當前點的parent進行比較并交換僻弹,直到滿足x >= queue[parent]為止阿浓。注意這里的比較可以是元素的自然順序,也可以是依靠比較器的順序蹋绽。

element()和peek()

element()peek()的語義完全相同芭毙,都是獲取但不刪除隊首元素,也就是隊列中權(quán)值最小的那個元素卸耘,二者唯一的區(qū)別是當方法失敗時前者拋出異常退敦,后者返回null。根據(jù)小頂堆的性質(zhì)蚣抗,堆頂那個元素就是全局最小的那個侈百;由于堆用數(shù)組表示,根據(jù)下標關(guān)系翰铡,0下標處的那個元素既是堆頂元素钝域。所以直接返回數(shù)組0下標處的那個元素即可

代碼也就非常簡潔:

//peek()
public E peek() {
    if (size == 0)
        return null;
    return (E) queue[0];//0下標處的那個元素就是最小的那個
}

remove()和poll()

remove()poll()方法的語義也完全相同锭魔,都是獲取并刪除隊首元素例证,區(qū)別是當方法失敗時前者拋出異常,后者返回null迷捧。由于刪除操作會改變隊列的結(jié)構(gòu)织咧,為維護小頂堆的性質(zhì),需要進行必要的調(diào)整党涕。

代碼如下:

public E poll() {
    if (size == 0)
        return null;
    int s = --size;
    modCount++;
    E result = (E) queue[0];//0下標處的那個元素就是最小的那個
    E x = (E) queue[s];
    queue[s] = null;
    if (s != 0)
        siftDown(0, x);//調(diào)整
    return result;
}

上述代碼首先記錄0下標處的元素烦感,并用最后一個元素替換0下標位置的元素巡社,之后調(diào)用siftDown()方法對堆進行調(diào)整膛堤,最后返回原來0下標處的那個元素(也就是最小的那個元素)。重點是siftDown(int k, E x)方法晌该,該方法的作用是從k指定的位置開始肥荔,將x逐層向下與當前點的左右孩子中較小的那個交換,直到x小于或等于左右孩子中的任何一個為止朝群。

//siftDown()
private void siftDown(int k, E x) {
    int half = size >>> 1;
    while (k < half) {
        //首先找到左右孩子中較小的那個燕耿,記錄到c里,并用child記錄其下標
        int child = (k << 1) + 1;//leftNo = parentNo*2+1
        Object c = queue[child];
        int right = child + 1;
        if (right < size &&
            comparator.compare((E) c, (E) queue[right]) > 0)
            c = queue[child = right];
        if (comparator.compare(x, (E) c) <= 0)
            break;
        queue[k] = c;//然后用c取代原來的值
        k = child;
    }
    queue[k] = x;
}

remove(Object o)

remove(Object o)方法用于刪除隊列中跟o相等的某一個元素(如果有多個相等姜胖,只刪除一個)誉帅,該方法不是Queue接口內(nèi)的方法,而是Collection接口的方法。由于刪除操作會改變隊列結(jié)構(gòu)蚜锨,所以要進行調(diào)整档插;又由于刪除元素的位置可能是任意的,所以調(diào)整過程比其它函數(shù)稍加繁瑣亚再。具體來說郭膛,remove(Object o)可以分為2種情況:1. 刪除的是最后一個元素。直接刪除即可氛悬,不需要調(diào)整则剃。2. 刪除的不是最后一個元素,從刪除點開始以最后一個元素為參照調(diào)用一次siftDown()即可如捅。此處不再贅述棍现。

具體代碼如下:

//remove(Object o)
public boolean remove(Object o) {
    //通過遍歷數(shù)組的方式找到第一個滿足o.equals(queue[i])元素的下標
    int i = indexOf(o);
    if (i == -1)
        return false;
    int s = --size;
    if (s == i) //情況1
        queue[i] = null;
    else {
        E moved = (E) queue[s];
        queue[s] = null;
        siftDown(i, moved);//情況2
        ......
    }
    return true;
}
最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市镜遣,隨后出現(xiàn)的幾起案子轴咱,更是在濱河造成了極大的恐慌,老刑警劉巖烈涮,帶你破解...
    沈念sama閱讀 211,194評論 6 490
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件朴肺,死亡現(xiàn)場離奇詭異,居然都是意外死亡坚洽,警方通過查閱死者的電腦和手機戈稿,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,058評論 2 385
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來讶舰,“玉大人鞍盗,你說我怎么就攤上這事√纾” “怎么了般甲?”我有些...
    開封第一講書人閱讀 156,780評論 0 346
  • 文/不壞的土叔 我叫張陵,是天一觀的道長鹅颊。 經(jīng)常有香客問我敷存,道長,這世上最難降的妖魔是什么堪伍? 我笑而不...
    開封第一講書人閱讀 56,388評論 1 283
  • 正文 為了忘掉前任锚烦,我火速辦了婚禮,結(jié)果婚禮上帝雇,老公的妹妹穿的比我還像新娘涮俄。我一直安慰自己,他們只是感情好尸闸,可當我...
    茶點故事閱讀 65,430評論 5 384
  • 文/花漫 我一把揭開白布彻亲。 她就那樣靜靜地躺著孕锄,像睡著了一般。 火紅的嫁衣襯著肌膚如雪苞尝。 梳的紋絲不亂的頭發(fā)上硫惕,一...
    開封第一講書人閱讀 49,764評論 1 290
  • 那天,我揣著相機與錄音野来,去河邊找鬼恼除。 笑死,一個胖子當著我的面吹牛曼氛,可吹牛的內(nèi)容都是我干的豁辉。 我是一名探鬼主播,決...
    沈念sama閱讀 38,907評論 3 406
  • 文/蒼蘭香墨 我猛地睜開眼舀患,長吁一口氣:“原來是場噩夢啊……” “哼徽级!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起聊浅,我...
    開封第一講書人閱讀 37,679評論 0 266
  • 序言:老撾萬榮一對情侶失蹤餐抢,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后低匙,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體旷痕,經(jīng)...
    沈念sama閱讀 44,122評論 1 303
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 36,459評論 2 325
  • 正文 我和宋清朗相戀三年顽冶,在試婚紗的時候發(fā)現(xiàn)自己被綠了欺抗。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 38,605評論 1 340
  • 序言:一個原本活蹦亂跳的男人離奇死亡强重,死狀恐怖绞呈,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情间景,我是刑警寧澤佃声,帶...
    沈念sama閱讀 34,270評論 4 329
  • 正文 年R本政府宣布甥温,位于F島的核電站疤苹,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏跑筝。R本人自食惡果不足惜碗誉,卻給世界環(huán)境...
    茶點故事閱讀 39,867評論 3 312
  • 文/蒙蒙 一召嘶、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧哮缺,春花似錦、人聲如沸甲喝。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,734評論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至糠溜,卻和暖如春淳玩,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背非竿。 一陣腳步聲響...
    開封第一講書人閱讀 31,961評論 1 265
  • 我被黑心中介騙來泰國打工蜕着, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人红柱。 一個月前我還...
    沈念sama閱讀 46,297評論 2 360
  • 正文 我出身青樓承匣,卻偏偏與公主長得像,于是被迫代替她去往敵國和親锤悄。 傳聞我的和親對象是個殘疾皇子韧骗,可洞房花燭夜當晚...
    茶點故事閱讀 43,472評論 2 348

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