Java 優(yōu)先級隊列 PriorityQueue

更多 Java 集合類方面的文章,請參見文集《Java 集合類》


優(yōu)先級隊列 PriorityQueue

PriorityQueue 類在 Java 1.5 中引入。

PriorityQueue 是基于優(yōu)先堆的一個無界隊列已脓,這個優(yōu)先隊列中的元素可以默認自然排序或者通過提供的
Comparator 在隊列實例化的時排序。

PriorityQueue 不允許空值通殃,而且不支持 non-comparable(不可比較)的對象度液,比如用戶自定義的類。優(yōu)先隊列要求使用 Java Comparable 和 Comparator 接口給對象排序画舌,并且在排序時會按照優(yōu)先級處理其中的元素堕担。

PriorityQueue 的大小是不受限制的,但在創(chuàng)建時可以指定初始大小曲聂。當(dāng)我們向優(yōu)先隊列增加元素的時候霹购,隊列大小會自動增加。

PriorityQueue 是非線程安全的朋腋,所以 Java 提供了 PriorityBlockingQueue(實現(xiàn) BlockingQueue接口)用于Java 多線程環(huán)境齐疙。

示例:

public class PriorityQueueTest {

    public static void main(String[] args) {
        Queue<Integer> queue1 = new PriorityQueue<Integer>();
        queue1.add(2);
        queue1.add(1);
        queue1.add(3);

        while (!queue1.isEmpty()) {
            Integer i = queue1.poll();
            System.out.println(i);
        }

        Comparator<Student> comparator = new Comparator<Student>() {
            @Override
            public int compare(Student o1, Student o2) {
                return (o1.id - o2.id);
            }
        };

        Queue<Student> queue2 = new PriorityQueue<Student>(comparator);
        queue2.add(new Student(2, "B"));
        queue2.add(new Student(1, "A"));
        queue2.add(new Student(3, "C"));

        while (!queue2.isEmpty()) {
            Student s = queue2.poll();
            System.out.println(s.toString());
        }
    }

    public static class Student {
        private int id;
        private String name;

        public Student(int id, String name) {
            this.id = id;
            this.name = name;
        }

        public String toString() {
            return id + "-" + name;
        }
    }
}

輸出:

1
2
3
1-A
2-B
3-C

優(yōu)先級隊列 PriorityQueue 的實現(xiàn)原理

通過堆實現(xiàn),具體說是通過完全二叉樹(complete binary tree)實現(xiàn)的小頂堆(任意一個非葉子節(jié)點的權(quán)值,都不大于其左右子節(jié)點的權(quán)值),也就意味著可以通過數(shù)組來作為 PriorityQueue 的底層實現(xiàn)睹耐。

    /**
     * Priority queue represented as a balanced binary heap: the two
     * children of queue[n] are queue[2*n+1] and queue[2*(n+1)].  The
     * priority queue is ordered by comparator, or by the elements'
     * natural ordering, if comparator is null: For each node n in the
     * heap and each descendant d of n, n <= d.  The element with the
     * lowest value is in queue[0], assuming the queue is nonempty.
     */
    transient Object[] queue; // non-private to simplify nested class access

    /**
     * The number of elements in the priority queue.
     */
    private int size = 0;

父子節(jié)點的編號之間有如下關(guān)系:

  • leftNo = parentNo * 2 + 1
  • rightNo = parentNo * 2 + 2
  • parentNo = (nodeNo - 1) / 2

add(E e)offer(E e) 操作的時間復(fù)雜度為 log(N):

    public boolean offer(E e) {
        if (e == 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);
        return true;
    }

poll() 操作的時間復(fù)雜度為 log(N):

    public E poll() {
        if (size == 0)
            return null;
        int s = --size;
        modCount++;
        E result = (E) queue[0];
        E x = (E) queue[s];
        queue[s] = null;
        if (s != 0)
            siftDown(0, x);
        return result;
    }

peek() 操作的時間復(fù)雜度為 O(1):

    public E peek() {
        return (size == 0) ? null : (E) queue[0];
    }

優(yōu)先級隊列 PriorityQueue 的使用場景

求 Top K 大的元素
維護數(shù)據(jù)流中的中位數(shù)


引用:
Java優(yōu)先隊列(PriorityQueue)示例
深入理解Java PriorityQueue

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末郁妈,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌济锄,老刑警劉巖绒净,帶你破解...
    沈念sama閱讀 217,734評論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件催训,死亡現(xiàn)場離奇詭異洽议,居然都是意外死亡,警方通過查閱死者的電腦和手機漫拭,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,931評論 3 394
  • 文/潘曉璐 我一進店門亚兄,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人采驻,你說我怎么就攤上這事审胚。” “怎么了礼旅?”我有些...
    開封第一講書人閱讀 164,133評論 0 354
  • 文/不壞的土叔 我叫張陵膳叨,是天一觀的道長。 經(jīng)常有香客問我痘系,道長菲嘴,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 58,532評論 1 293
  • 正文 為了忘掉前任汰翠,我火速辦了婚禮龄坪,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘复唤。我一直安慰自己健田,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 67,585評論 6 392
  • 文/花漫 我一把揭開白布佛纫。 她就那樣靜靜地躺著妓局,像睡著了一般。 火紅的嫁衣襯著肌膚如雪雳旅。 梳的紋絲不亂的頭發(fā)上跟磨,一...
    開封第一講書人閱讀 51,462評論 1 302
  • 那天间聊,我揣著相機與錄音攒盈,去河邊找鬼。 笑死哎榴,一個胖子當(dāng)著我的面吹牛型豁,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播尚蝌,決...
    沈念sama閱讀 40,262評論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼迎变,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了飘言?” 一聲冷哼從身側(cè)響起衣形,我...
    開封第一講書人閱讀 39,153評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后谆吴,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體倒源,經(jīng)...
    沈念sama閱讀 45,587評論 1 314
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,792評論 3 336
  • 正文 我和宋清朗相戀三年句狼,在試婚紗的時候發(fā)現(xiàn)自己被綠了笋熬。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 39,919評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡腻菇,死狀恐怖胳螟,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情筹吐,我是刑警寧澤糖耸,帶...
    沈念sama閱讀 35,635評論 5 345
  • 正文 年R本政府宣布,位于F島的核電站丘薛,受9級特大地震影響蔬捷,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜榔袋,卻給世界環(huán)境...
    茶點故事閱讀 41,237評論 3 329
  • 文/蒙蒙 一周拐、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧凰兑,春花似錦妥粟、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,855評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至锅知,卻和暖如春播急,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背售睹。 一陣腳步聲響...
    開封第一講書人閱讀 32,983評論 1 269
  • 我被黑心中介騙來泰國打工桩警, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人昌妹。 一個月前我還...
    沈念sama閱讀 48,048評論 3 370
  • 正文 我出身青樓捶枢,卻偏偏與公主長得像,于是被迫代替她去往敵國和親飞崖。 傳聞我的和親對象是個殘疾皇子烂叔,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 44,864評論 2 354

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