日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

PriorityQueue源码解析

發(fā)布時(shí)間:2025/3/15 编程问答 19 豆豆
生活随笔 收集整理的這篇文章主要介紹了 PriorityQueue源码解析 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

一、概述

優(yōu)先隊(duì)列的作用是能保證每次取出的元素都是隊(duì)列中權(quán)值最小的(Java的優(yōu)先隊(duì)列每次取最小元素,C++的優(yōu)先隊(duì)列每次取最大元素)。這里牽涉到了大小關(guān)系,元素大小的評判可以通過元素本身的自然順序(natural ordering),也可以通過構(gòu)造時(shí)傳入的比較器(Comparator,類似于C++的仿函數(shù))。
Java中PriorityQueue實(shí)現(xiàn)了Queue接口,不允許放入null元素;其通過堆實(shí)現(xiàn),具體說是通過完全二叉樹(complete binary tree)實(shí)現(xiàn)的小頂堆(任意一個(gè)非葉子節(jié)點(diǎn)的權(quán)值,都不大于其左右子節(jié)點(diǎn)的權(quán)值),也就意味著可以通過數(shù)組來作為PriorityQueue的底層實(shí)現(xiàn)。

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

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

通過上述三個(gè)公式,可以輕易計(jì)算出某個(gè)節(jié)點(diǎn)的父節(jié)點(diǎn)以及子節(jié)點(diǎn)的下標(biāo)。這也就是為什么可以直接用數(shù)組來存儲堆的原因。

PriorityQueue的peek()和element操作是常數(shù)時(shí)間,add(), offer(), 無參數(shù)的remove()以及poll()方法的時(shí)間復(fù)雜度都是log(N)。

二、 方法剖析

  • add()和offer()
    add(E e)和offer(E e)的語義相同,都是向優(yōu)先隊(duì)列中插入元素,只是Queue接口規(guī)定二者對插入失敗時(shí)的處理不同,前者在插入失敗時(shí)拋出異常,后則則會返回false。對于PriorityQueue這兩個(gè)方法其實(shí)沒什么差別。
  • 新加入的元素可能會破壞小頂堆的性質(zhì),因此需要進(jìn)行必要的調(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);//自動擴(kuò)容size = i + 1;if (i == 0)//隊(duì)列原來為空,這是插入的第一個(gè)元素queue[0] = e;elsesiftUp(i, e);//調(diào)整return true; }

    上述代碼中,擴(kuò)容函數(shù)grow()類似于ArrayList里的grow()函數(shù),就是再申請一個(gè)更大的數(shù)組,并將原數(shù)組的元素復(fù)制過去,這里不再贅述。需要注意的是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)/2Object e = queue[parent];if (comparator.compare(x, (E) e) >= 0)//調(diào)用比較器的比較方法break;queue[k] = e;k = parent;}queue[k] = x; }

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

  • element()和peek()
    element()和peek()的語義完全相同,都是獲取但不刪除隊(duì)首元素,也就是隊(duì)列中權(quán)值最小的那個(gè)元素,二者唯一的區(qū)別是當(dāng)方法失敗時(shí)前者拋出異常,后者返回null。根據(jù)小頂堆的性質(zhì),堆頂那個(gè)元素就是全局最小的那個(gè);由于堆用數(shù)組表示,根據(jù)下標(biāo)關(guān)系,0下標(biāo)處的那個(gè)元素既是堆頂元素。所以直接返回?cái)?shù)組0下標(biāo)處的那個(gè)元素即可。
  • 代碼也就非常簡潔: //peek()

    public E peek() {if (size == 0)return null;return (E) queue[0];//0下標(biāo)處的那個(gè)元素就是最小的那個(gè) }
  • remove()和poll()
    remove()和poll()方法的語義也完全相同,都是獲取并刪除隊(duì)首元素,區(qū)別是當(dāng)方法失敗時(shí)前者拋出異常,后者返回null。由于刪除操作會改變隊(duì)列的結(jié)構(gòu),為維護(hù)小頂堆的性質(zhì),需要進(jìn)行必要的調(diào)整。
  • 代碼如下:

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

    上述代碼首先記錄0下標(biāo)處的元素,并用最后一個(gè)元素替換0下標(biāo)位置的元素,之后調(diào)用siftDown()方法對堆進(jìn)行調(diào)整,最后返回原來0下標(biāo)處的那個(gè)元素(也就是最小的那個(gè)元素)。重點(diǎn)是siftDown(int k, E x)方法,該方法的作用是從k指定的位置開始,將x逐層向下與當(dāng)前點(diǎn)的左右孩子中較小的那個(gè)交換,直到x小于或等于左右孩子中的任何一個(gè)為止。

    //siftDown() private void siftDown(int k, E x) {int half = size >>> 1;while (k < half) {//首先找到左右孩子中較小的那個(gè),記錄到c里,并用child記錄其下標(biāo)int child = (k << 1) + 1;//leftNo = parentNo*2+1Object 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)方法用于刪除隊(duì)列中跟o相等的某一個(gè)元素(如果有多個(gè)相等,只刪除一個(gè)),該方法不是Queue接口內(nèi)的方法,而是Collection接口的方法。由于刪除操作會改變隊(duì)列結(jié)構(gòu),所以要進(jìn)行調(diào)整;又由于刪除元素的位置可能是任意的,所以調(diào)整過程比其它函數(shù)稍加繁瑣。具體來說,remove(Object o)可以分為2種情況:

  • 刪除的是最后一個(gè)元素。直接刪除即可,不需要調(diào)整。
  • 刪除的不是最后一個(gè)元素,從刪除點(diǎn)開始以最后一個(gè)元素為參照調(diào)用一次siftDown()即可。此處不再贅述。

  • 具體代碼如下:

    //remove(Object o) public boolean remove(Object o) {//通過遍歷數(shù)組的方式找到第一個(gè)滿足o.equals(queue[i])元素的下標(biāo)int i = indexOf(o);if (i == -1)return false;int s = --size;if (s == i) //情況1queue[i] = null;else {E moved = (E) queue[s];queue[s] = null;siftDown(i, moved);//情況2......}return true; }

    文章轉(zhuǎn)自

    總結(jié)

    以上是生活随笔為你收集整理的PriorityQueue源码解析的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

    如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

    主站蜘蛛池模板: 欧美xxxx少妇 | 亚洲成人生活片 | 精品乱子伦一区二区三区 | 日韩欧美中文在线 | 特黄三级又爽又粗又大 | 久草大| 日本成人综合 | 午夜福利123| 亚洲天堂免费在线 | 美国色视频| 国产又粗又猛又爽又黄的视频一 | 日韩欧美视频在线播放 | av自拍一区| 国产精品国产三级国产普通话蜜臀 | 久久亚洲AV成人无码一二三 | 最新av导航 | 最新av网址在线观看 | 求个黄色网址 | 狠狠干中文字幕 | 日本国产三级xxxxxx | 中文理论片 | 青青操网站 | 国产一级在线视频 | 欧美激情va永久在线播放 | 亚洲高清在线 | 亚洲国产视频一区二区三区 | 免费黄色观看 | 欧美成人三级在线视频 | 国产精品无码免费在线观看 | 中文字幕在线网站 | 中文字幕人妻一区二 | 亚洲少妇第一页 | 亚洲男女在线观看 | 嫩草www| 九九热精品在线 | 日韩黄色在线 | 日本人做受免费视频 | 亚州av在线播放 | 欧美熟妇精品一区二区蜜桃视频 | 69久久久久 | 五十路中出| 国产激情成人 | 日韩在线视频网站 | av中文字幕第一页 | 欧美男女交配视频 | 涩涩涩综合 | 国产欧美亚洲精品 | 四虎福利视频 | 校园春色亚洲 | 欧美三级在线观看视频 | 嫩草影院菊竹影院 | 美女狂揉羞羞的视频 | 亚洲男人天堂2017 | 婷婷国产一区二区三区 | 亚洲一级av无码毛片精品 | 激情 小说 亚洲 图片 伦 | 日韩在线91 | 婷综合 | 亚洲精品一区二区二区 | 五月婷婷天 | 久久91av | 久久综合免费视频 | 在线免费看mv的网站入口 | 毛片大全免费看 | 黄色3级视频 | 美国三级a三级18 | 国产三级福利 | 亚洲最大福利视频网 | 青青草手机在线视频 | 一级啪啪片 | 国产精品无码久久久久久电影 | 人妻无码中文久久久久专区 | 99re在线精品视频 | 久久精品久久久久久久 | 6080电视影片在线观看 | 性欧美高清 | 日韩av福利| 国产精品天天狠天天看 | 激情影音 | 99成人在线 | 国产精品主播视频 | 熟女一区二区三区四区 | 欧美性69 | 日本专区在线 | 日本在线观看免费 | 91人妻一区二区三区蜜臀 | 婷婷激情久久 | 国产成人精品二区三区亚瑟 | 6080黄色 | 色综合91 | 热久久精 | 一本色道久久综合狠狠躁 | 久久久天堂国产精品女人 | 国产精品成av人在线视午夜片 | 国产精品一区二区久久毛片 | 一级aaaa毛片 | 婷婷色九月 | 潘金莲黄色一级片 | 少妇av网 |