Java集合之優(yōu)先隊(duì)列PriorityQueue

PriorityQueue

源自java.util.PriorityQueue系谐,繼承結(jié)構(gòu):

java.lang.Object
    java.util.AbstractCollection<E>
        java.util.AbstractQueue<E>
            java.util.PriorityQueue<E>

PriorityQueue也就是優(yōu)先隊(duì)列启泣,優(yōu)先隊(duì)列里面的元素是有序的(元素本身自然有序,或者通過Comparator能夠進(jìn)行比較),元素不允許為null值或者不可比較的值。隊(duì)列的頭是所有元素中最小的,如果有多個(gè)袄琳,則返回其中一個(gè)(任意的窿凤,即不穩(wěn)定排序)。iterator()方法返回的隊(duì)列的迭代器跨蟹,需要注意的是迭代器返回的元素是無序雳殊。PriorityQueue不是線程安全的,如果需要線程安全的隊(duì)列窗轩,可以使用PriorityBlockingQueue

代碼

優(yōu)先隊(duì)列是使用數(shù)組來實(shí)現(xiàn)的夯秃,并用來表示堆的邏輯接口。類中主要的屬性:

// 當(dāng)前節(jié)點(diǎn)的左孩子為2n+1痢艺,右孩子為2n+2
transient Object[] queue
// 隊(duì)列的大小仓洼,可以在構(gòu)造方法中傳遞
private int size = 0;
// modCount用來記錄隊(duì)列修改的次數(shù)
transient int modCount = 0;

相關(guān)方法:

add: 隊(duì)列中添加元素

public boolean add(E e) {
    return offer(e);
}

可以看到直接調(diào)用的是offer方法,offer也是給隊(duì)列中添加元素:

public boolean offer(E e) {
    if (e == null)
        throw new NullPointerException();
    modCount++;
    int i = size;
    if (i >= queue.length)
        //隊(duì)列擴(kuò)容,度列長度小于64直接double堤舒,大于64色建,增長50%
        // 隊(duì)列的最大容量為Integer.MAX_VALUE - 8
        // 多余的8個(gè)用來針對有些虛擬機(jī)需要保留額外的信息
        grow(i + 1);
    size = i + 1;
    if (i == 0)
        queue[0] = e;
    else
        // 上浮操作
        siftUp(i, e);
    return true;
}

private void siftUpComparable(int k, E x) {
    Comparable<? super E> key = (Comparable<? super E>) x;
    while (k > 0) {
        // 找到父親節(jié)點(diǎn)
        int parent = (k - 1) >>> 1;
        Object e = queue[parent];
        // 如果當(dāng)前節(jié)點(diǎn)比父親節(jié)點(diǎn)大,停止查找舌缤,表示位置是合適的
        // 如果小箕戳,則繼續(xù)向上查找,并且邏輯上交換一下元素
        if (key.compareTo((E) e) >= 0)
            break;
        // k位置放父親節(jié)點(diǎn)
        queue[k] = e;
        // 從父親節(jié)點(diǎn)的位置繼續(xù)向上找
        k = parent;
    }
    queue[k] = key;
}

另一個(gè)比較關(guān)鍵的方法是poll国撵,用來從堆中彈出元素:

public E poll() {
    if (size == 0)
        return null;
    int s = --size;
    modCount++;
    // 每次都是返回第一個(gè)元素
    E result = (E) queue[0];
    // 獲取最后一個(gè)元素
    E x = (E) queue[s];
    // 將最后一個(gè)元素置為空
    queue[s] = null;
    if (s != 0)
        // 下沉操作陵吸,將x放在堆的頭部,然后下沉和合適的位置
        siftDown(0, x);
    return result;
}

private void siftDown(int k, E x) {
    if (comparator != null)
        siftDownUsingComparator(k, x);
    else
        siftDownComparable(k, x);
}
private void siftDownComparable(int k, E x) {
    Comparable<? super E> key = (Comparable<? super E>)x;
    int half = size >>> 1;        // loop while a non-leaf
    // 以隊(duì)列大小的一半為邊界介牙,因?yàn)槭呛妥庸?jié)點(diǎn)交換壮虫,超過一半就沒有子節(jié)點(diǎn)了
    while (k < half) {
        //找到孩子節(jié)點(diǎn),如果孩子節(jié)點(diǎn)小于自己环础,就交換
        // 如果兩個(gè)孩子都小于自己囚似,交換右節(jié)點(diǎn)
        int child = (k << 1) + 1; // assume left child is least
        Object c = queue[child];
        int right = child + 1;
        if (right < size &&
            ((Comparable<? super E>) c).compareTo((E) queue[right]) > 0)
            c = queue[child = right];
        if (key.compareTo((E) c) <= 0)
            break;
        queue[k] = c;
        k = child;
    }
    queue[k] = key;
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個(gè)濱河市线得,隨后出現(xiàn)的幾起案子饶唤,更是在濱河造成了極大的恐慌,老刑警劉巖框都,帶你破解...
    沈念sama閱讀 218,941評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件搬素,死亡現(xiàn)場離奇詭異呵晨,居然都是意外死亡魏保,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,397評論 3 395
  • 文/潘曉璐 我一進(jìn)店門摸屠,熙熙樓的掌柜王于貴愁眉苦臉地迎上來谓罗,“玉大人,你說我怎么就攤上這事季二¢菰郏” “怎么了揭措?”我有些...
    開封第一講書人閱讀 165,345評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長刻蚯。 經(jīng)常有香客問我绊含,道長,這世上最難降的妖魔是什么炊汹? 我笑而不...
    開封第一講書人閱讀 58,851評論 1 295
  • 正文 為了忘掉前任躬充,我火速辦了婚禮,結(jié)果婚禮上讨便,老公的妹妹穿的比我還像新娘充甚。我一直安慰自己,他們只是感情好霸褒,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,868評論 6 392
  • 文/花漫 我一把揭開白布伴找。 她就那樣靜靜地躺著,像睡著了一般废菱。 火紅的嫁衣襯著肌膚如雪技矮。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,688評論 1 305
  • 那天殊轴,我揣著相機(jī)與錄音穆役,去河邊找鬼。 笑死梳凛,一個(gè)胖子當(dāng)著我的面吹牛耿币,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播韧拒,決...
    沈念sama閱讀 40,414評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼淹接,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了叛溢?” 一聲冷哼從身側(cè)響起塑悼,我...
    開封第一講書人閱讀 39,319評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎楷掉,沒想到半個(gè)月后厢蒜,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,775評論 1 315
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡烹植,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,945評論 3 336
  • 正文 我和宋清朗相戀三年斑鸦,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片草雕。...
    茶點(diǎn)故事閱讀 40,096評論 1 350
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡巷屿,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出墩虹,到底是詐尸還是另有隱情嘱巾,我是刑警寧澤憨琳,帶...
    沈念sama閱讀 35,789評論 5 346
  • 正文 年R本政府宣布,位于F島的核電站旬昭,受9級特大地震影響篙螟,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜问拘,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,437評論 3 331
  • 文/蒙蒙 一闲擦、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧场梆,春花似錦墅冷、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,993評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至顶岸,卻和暖如春腔彰,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背辖佣。 一陣腳步聲響...
    開封第一講書人閱讀 33,107評論 1 271
  • 我被黑心中介騙來泰國打工霹抛, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留,地道東北人卷谈。 一個(gè)月前我還...
    沈念sama閱讀 48,308評論 3 372
  • 正文 我出身青樓杯拐,卻偏偏與公主長得像,于是被迫代替她去往敵國和親世蔗。 傳聞我的和親對象是個(gè)殘疾皇子端逼,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 45,037評論 2 355