日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

詳解JAVA中priorityqueue的具體使用

瀏覽:35日期:2022-08-18 08:51:51

Java中PriorityQueue通過二叉小頂堆實(shí)現(xiàn),可以用一棵完全二叉樹表示。本文從Queue接口函數(shù)出發(fā),結(jié)合生動(dòng)的圖解,深入淺出地分析PriorityQueue每個(gè)操作的具體過程和時(shí)間復(fù)雜度,將讓讀者建立對(duì)PriorityQueue建立清晰而深入的認(rèn)識(shí)。

總體介紹

前面以JavaArrayDeque為例講解了Stack和Queue,其實(shí)還有一種特殊的隊(duì)列叫做PriorityQueue,即優(yōu)先隊(duì)列。優(yōu)先隊(duì)列的作用是能保證每次取出的元素都是隊(duì)列中權(quán)值最小的(Java的優(yōu)先隊(duì)列每次取最小元素,C++的優(yōu)先隊(duì)列每次取最大元素)。這里牽涉到了大小關(guān)系,元素大小的評(píng)判可以通過元素本身的自然順序(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)。

詳解JAVA中priorityqueue的具體使用

上圖中我們給每個(gè)元素按照層序遍歷的方式進(jìn)行了編號(hào),如果你足夠細(xì)心,會(huì)發(fā)現(xiàn)父節(jié)點(diǎn)和子節(jié)點(diǎn)的編號(hào)是有聯(lián)系的,更確切的說父子節(jié)點(diǎn)的編號(hào)之間有如下關(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ù)組來存儲(chǔ)堆的原因。

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

方法剖析

add()和offer()

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

詳解JAVA中priorityqueue的具體使用

新加入的元素可能會(huì)破壞小頂堆的性質(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);//自動(dòng)擴(kuò)容 size = i + 1; if (i == 0)//隊(duì)列原來為空,這是插入的第一個(gè)元素 queue[0] = e; else siftUp(i, e);//調(diào)整 return true;}

上述代碼中,擴(kuò)容函數(shù)grow()類似于ArrayList里的grow()函數(shù),就是再申請(qǐng)一個(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)/2 Object e = queue[parent]; if (comparator.compare(x, (E) e) >= 0)//調(diào)用比較器的比較方法 break; queue[k] = e; k = parent; } queue[k] = x;}

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

element()和peek()

element()和peek()的語(yǔ)義完全相同,都是獲取但不刪除隊(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è)元素即可。

詳解JAVA中priorityqueue的具體使用

代碼也就非常簡(jiǎn)潔:

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

remove()和poll()

remove()和poll()方法的語(yǔ)義也完全相同,都是獲取并刪除隊(duì)首元素,區(qū)別是當(dāng)方法失敗時(shí)前者拋出異常,后者返回null。由于刪除操作會(huì)改變隊(duì)列的結(jié)構(gòu),為維護(hù)小頂堆的性質(zhì),需要進(jìn)行必要的調(diào)整。

詳解JAVA中priorityqueue的具體使用

代碼如下:

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()方法對(duì)堆進(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+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)方法用于刪除隊(duì)列中跟o相等的某一個(gè)元素(如果有多個(gè)相等,只刪除一個(gè)),該方法不是Queue接口內(nèi)的方法,而是Collection接口的方法。由于刪除操作會(huì)改變隊(duì)列結(jié)構(gòu),所以要進(jìn)行調(diào)整;又由于刪除元素的位置可能是任意的,所以調(diào)整過程比其它函數(shù)稍加繁瑣。具體來說,remove(Object o)可以分為2種情況:1. 刪除的是最后一個(gè)元素。直接刪除即可,不需要調(diào)整。2. 刪除的不是最后一個(gè)元素,從刪除點(diǎn)開始以最后一個(gè)元素為參照調(diào)用一次siftDown()即可。此處不再贅述。

詳解JAVA中priorityqueue的具體使用

具體代碼如下:

//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) //情況1 queue[i] = null; else { E moved = (E) queue[s]; queue[s] = null; siftDown(i, moved);//情況2 ...... } return true;}

到此這篇關(guān)于詳解JAVA中priorityqueue的具體使用的文章就介紹到這了,更多相關(guān)JAVA priorityqueue內(nèi)容請(qǐng)搜索好吧啦網(wǎng)以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持好吧啦網(wǎng)!

標(biāo)簽: Java
相關(guān)文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产婷婷精品| 午夜精品免费| 欧美一区自拍| 日韩亚洲精品在线观看| 日本a口亚洲| 国产精品亚洲欧美日韩一区在线| 欧美一区自拍| 国产欧美大片| 91精品日本| 国产精品v日韩精品v欧美精品网站 | 国产亚洲久久| 国产精品一区二区美女视频免费看 | 国产日韩欧美一区在线| 国产精品黄色| 日韩欧美综合| 婷婷中文字幕一区| 亚洲欧美久久精品| 国产精久久一区二区| 国产黄大片在线观看| 激情婷婷亚洲| 免费在线观看成人| 国产欧美久久一区二区三区| 国产福利片在线观看| 欧美日韩国产免费观看| 亚洲免费福利一区| 国产一卡不卡| 午夜精品成人av| 日韩精品一卡二卡三卡四卡无卡| 日本aⅴ亚洲精品中文乱码| 精品一区二区三区中文字幕| 在线视频观看日韩| 日本不卡高清| 久久青青视频| 中文一区一区三区免费在线观 | 欧美中文一区二区| 蜜桃久久精品一区二区| 国产精品麻豆成人av电影艾秋| 98精品久久久久久久| 亚洲一区不卡| 国产精品伦一区二区| 成人日韩在线| 亚洲精品一二三**| 加勒比视频一区| 久久福利毛片| 国产成人精选| 免费看欧美美女黄的网站| 欧美黑人巨大videos精品| 国产精品av久久久久久麻豆网| 青青草国产成人99久久| 午夜精品成人av| 亚洲精品大全| 日韩亚洲一区在线| 亚洲精品进入| 桃色一区二区| 日本午夜精品一区二区三区电影| 免费高潮视频95在线观看网站| 亚洲久久视频| 久久精品国内一区二区三区水蜜桃| 日韩中文字幕无砖| 亚洲一级少妇| 欧美偷窥清纯综合图区| 久久黄色影院| 国产精品久久久久77777丨| 国产精品88久久久久久| 国产福利一区二区三区在线播放| 精品在线播放| 成人在线观看免费视频| 日韩欧美三区| 亚洲精品一区二区妖精| 国产一区二区三区不卡av| 亚洲精品系列| 91成人精品| 中文字幕在线看片| 国产精品成人3p一区二区三区| 免播放器亚洲| 99久久精品费精品国产| 精品久久91| 国产欧美精品| 亚洲一二三区视频| 亚洲电影在线一区二区三区| 精品资源在线| 日韩二区三区在线观看| 免费精品视频| 久久九九国产| 色欧美自拍视频| 久久福利在线| 久久精品72免费观看| 蜜桃久久久久久| 夜夜嗨网站十八久久| 99久久九九| 欧美日韩免费观看视频| 精品一区不卡| 国产福利资源一区| 国产日韩三级| 欧美日本精品| 亚洲精品少妇| 最近国产精品视频| 男女男精品网站| 99国产精品久久久久久久 | 免费欧美一区| 91精品亚洲| 成人片免费看| 成人亚洲一区| 国产成人a视频高清在线观看| 欧美激情日韩| 欧美国产视频| 久久亚洲人体| 精品国产中文字幕第一页| 久久精品99国产精品| 亚洲视频电影在线| 免费精品视频最新在线| 视频一区中文字幕国产| 丝袜亚洲精品中文字幕一区| 性一交一乱一区二区洋洋av| 亚洲一区网站| 久久不射中文字幕| 另类国产ts人妖高潮视频| 在线视频亚洲| 亚洲欧美日本国产专区一区| 99在线精品视频在线观看| 亚洲一区二区三区高清不卡| 久久av在线| 亚洲精品一二| 日韩欧美三区| 国产欧美一区二区三区国产幕精品| 国产美女久久| 久久精品亚洲一区二区| 黄色精品视频| 在线一区视频观看| 亚洲男女av一区二区| 丝袜亚洲精品中文字幕一区| 亚洲精品影视| 国产精品对白久久久久粗| 老司机精品视频在线播放| 久久精品欧洲| 日韩成人亚洲| 亚洲少妇诱惑| 日本不卡视频在线观看| 国产探花在线精品| 久久久久亚洲精品中文字幕| 久久uomeier| 亚洲欧美日韩高清在线| 久久国产精品久久w女人spa| 日本不卡视频在线| 欧美交a欧美精品喷水| 黑森林国产精品av| av亚洲在线观看| 一区二区精品| 欧美激情一区| 色婷婷狠狠五月综合天色拍| 日韩午夜一区| 日本不卡不码高清免费观看| 国产精品nxnn| 日韩久久视频| 久热综合在线亚洲精品| 国产一级成人av| 日韩av免费| 亚洲三级网站| 韩日一区二区| 狠狠色综合网| 国产日韩一区二区三区在线播放| 激情国产在线| 视频在线观看国产精品| 国产精品久久久久久久免费软件| 综合日韩av| 99久久www免费| 日韩国产在线不卡视频| 国产成人精品免费视| 欧美福利专区| 91精品一区| 久久视频精品| 91福利精品在线观看| 群体交乱之放荡娇妻一区二区| 亚洲精一区二区三区| 欧美国产美女| 中文一区一区三区免费在线观 | 超级白嫩亚洲国产第一| 羞羞答答国产精品www一本| 欧美激情精品| 久久不射网站| 欧美国产小视频| 免费在线观看视频一区| 成人影视亚洲图片在线| 亚洲精品亚洲人成在线观看| 日本不卡免费高清视频在线| 日韩欧美精品一区二区综合视频| 中文在线а√天堂| 日韩成人一级| 欧美日韩日本国产亚洲在线| 国产精品久久久久久久免费软件| 亚洲精品99| 久久精品毛片| 中文视频一区| 精品欧美一区二区三区在线观看| 欧美一级一区| 午夜视频精品| 精品高清久久| 亚洲精品第一| 欧美日韩国产一区精品一区| 精品视频亚洲|