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

您的位置:首頁技術文章
文章詳情頁

詳解JAVA中priorityqueue的具體使用

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

Java中PriorityQueue通過二叉小頂堆實現,可以用一棵完全二叉樹表示。本文從Queue接口函數出發,結合生動的圖解,深入淺出地分析PriorityQueue每個操作的具體過程和時間復雜度,將讓讀者建立對PriorityQueue建立清晰而深入的認識。

總體介紹

前面以JavaArrayDeque為例講解了Stack和Queue,其實還有一種特殊的隊列叫做PriorityQueue,即優先隊列。優先隊列的作用是能保證每次取出的元素都是隊列中權值最小的(Java的優先隊列每次取最小元素,C++的優先隊列每次取最大元素)。這里牽涉到了大小關系,元素大小的評判可以通過元素本身的自然順序(natural ordering),也可以通過構造時傳入的比較器(Comparator,類似于C++的仿函數)。

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

詳解JAVA中priorityqueue的具體使用

上圖中我們給每個元素按照層序遍歷的方式進行了編號,如果你足夠細心,會發現父節點和子節點的編號是有聯系的,更確切的說父子節點的編號之間有如下關系:

leftNo = parentNo*2+1

rightNo = parentNo*2+2

parentNo = (nodeNo-1)/2

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

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

方法剖析

add()和offer()

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

詳解JAVA中priorityqueue的具體使用

新加入的元素可能會破壞小頂堆的性質,因此需要進行必要的調整。

//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);//調整 return true;}

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

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

element()和peek()

element()和peek()的語義完全相同,都是獲取但不刪除隊首元素,也就是隊列中權值最小的那個元素,二者唯一的區別是當方法失敗時前者拋出異常,后者返回null。根據小頂堆的性質,堆頂那個元素就是全局最小的那個;由于堆用數組表示,根據下標關系,0下標處的那個元素既是堆頂元素。所以直接返回數組0下標處的那個元素即可。

詳解JAVA中priorityqueue的具體使用

代碼也就非常簡潔:

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

remove()和poll()

remove()和poll()方法的語義也完全相同,都是獲取并刪除隊首元素,區別是當方法失敗時前者拋出異常,后者返回null。由于刪除操作會改變隊列的結構,為維護小頂堆的性質,需要進行必要的調整。

詳解JAVA中priorityqueue的具體使用

代碼如下:

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);//調整 return result;}

上述代碼首先記錄0下標處的元素,并用最后一個元素替換0下標位置的元素,之后調用siftDown()方法對堆進行調整,最后返回原來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接口內的方法,而是Collection接口的方法。由于刪除操作會改變隊列結構,所以要進行調整;又由于刪除元素的位置可能是任意的,所以調整過程比其它函數稍加繁瑣。具體來說,remove(Object o)可以分為2種情況:1. 刪除的是最后一個元素。直接刪除即可,不需要調整。2. 刪除的不是最后一個元素,從刪除點開始以最后一個元素為參照調用一次siftDown()即可。此處不再贅述。

詳解JAVA中priorityqueue的具體使用

具體代碼如下:

//remove(Object o)public boolean remove(Object o) { //通過遍歷數組的方式找到第一個滿足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;}

到此這篇關于詳解JAVA中priorityqueue的具體使用的文章就介紹到這了,更多相關JAVA priorityqueue內容請搜索好吧啦網以前的文章或繼續瀏覽下面的相關文章希望大家以后多多支持好吧啦網!

標簽: Java
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
蜜桃一区二区三区在线| 日韩视频1区| 国产精品视频一区二区三区综合| 日韩精品一卡二卡三卡四卡无卡| 亚洲一区亚洲| 亚洲深夜影院| 免费不卡在线观看| 综合激情一区| 免费成人在线视频观看| 免费一区二区视频| 最新国产精品| 日韩av中文在线观看| 欧美激情亚洲| 日本不良网站在线观看| 91成人精品视频| 亚洲福利一区| 精品久久91| 国产一区二区三区四区五区| 国产一区二区三区91| 日产精品一区二区| 中文字幕在线免费观看视频| 秋霞影院一区二区三区| 亚洲高清影视| 999在线观看精品免费不卡网站| 亚洲免费网址| 欧美精品国产白浆久久久久| 精品视频自拍| 国产99久久| 一区二区精品| 久久精品国产99国产| www.com.cn成人| 亚洲激情欧美| 青青草91视频| 另类专区亚洲| 久久亚洲电影| 国产精品最新| 99成人在线视频| 亚洲日韩视频| 麻豆精品av| 五月天久久久| 国产精品久久久网站| 99久久激情| 中文字幕日韩高清在线| 黄色欧美在线| 欧美交a欧美精品喷水| 欧美成人精品午夜一区二区| 日韩av首页| 视频一区中文字幕精品| 久久精品三级| 伊人久久亚洲热| 国产日产一区| 欧美精品高清| 亚洲人亚洲人色久| 国产精品嫩模av在线| 久久美女性网| 欧美中文一区| 99久久精品网站| 日本h片久久| 精品中文字幕一区二区三区av| 亚洲伊人精品酒店| 最新中文字幕在线播放| 亚洲精品乱码| 久久精品一区二区不卡| 国产日韩欧美| 亚洲深夜影院| 激情国产在线| 国产图片一区| 亚洲免费中文| 久久精品国产免费| 亚洲一级在线| 蜜桃成人精品| 国产日韩欧美一区二区三区在线观看 | 黄色成人91| 日韩在线视频一区二区三区| 国产精品蜜芽在线观看| 天堂久久av| 国内精品99| 久久免费视频66| 久久成人国产| 国产黄大片在线观看| 亚洲欧洲国产精品一区| 久久国产日本精品| 久久伊人亚洲| 日韩激情综合| 国产毛片一区| 亚洲国产综合在线看不卡| 精品网站aaa| 91成人福利| 免费在线观看精品| 亚洲香蕉网站| 欧美激情另类| 麻豆国产一区| 日韩精品视频在线看| 欧美午夜不卡| 日本免费久久| 精品视频在线观看网站| 国产亚洲电影| 日本午夜精品久久久| 日韩中文字幕一区二区三区| 99久久99视频只有精品| 91亚洲成人| 亚洲精品国产嫩草在线观看| 亚洲理论在线| 久久久夜夜夜| 国产精品一区二区精品视频观看| 男人的天堂久久精品| 99日韩精品| 欧美一区二区三区高清视频| 国产精品亚洲成在人线| 亚洲91网站| 蜜臀av性久久久久蜜臀aⅴ流畅| 亚洲一区二区毛片| 九九久久婷婷| 亚洲一级黄色| 99精品美女| 伊人精品一区| 99精品小视频| 欧美日韩一区二区综合| 日韩毛片视频| 免费福利视频一区二区三区| 在线天堂中文资源最新版| 神马午夜在线视频| 麻豆精品蜜桃| 伊伊综合在线| 中文字幕在线免费观看视频| 欧美男人天堂| 成人看片网站| 在线成人动漫av| 亚洲少妇诱惑| 水野朝阳av一区二区三区| 丝袜亚洲另类欧美| 日韩中文字幕1| 亚洲精品九九| 欧美私人啪啪vps| 香蕉人人精品| 欧美日韩激情| 日本精品在线中文字幕| 六月天综合网| 精品一二三区| 国产一区二区三区四区二区| 国产精品yjizz视频网| 精品捆绑调教一区二区三区| 欧美影院三区| 久久亚洲影院| 天堂va在线高清一区| 国产精品99久久免费| 超碰成人av| 99久久久久国产精品| 亚洲精品va| 最新国产精品| 日韩av一区二| 国产一区三区在线播放| 欧美丰满日韩| 欧美13videosex性极品| 久久国产电影| 人人精品人人爱| 欧美综合社区国产| 成人在线免费观看网站| 婷婷综合六月| 99pao成人国产永久免费视频 | 亚洲作爱视频| 日韩精品欧美大片| 中文字幕一区二区三区四区久久| 老司机精品视频网| 婷婷综合福利| 免费日韩成人| sm捆绑调教国产免费网站在线观看 | 国产三级精品三级在线观看国产| 国产欧美一区二区三区精品酒店| 香蕉国产精品| 日本成人在线不卡视频| 精品国产18久久久久久二百| 国产一区欧美| 日韩精品成人| 日韩免费高清| 亚洲欧洲国产精品一区| 红杏一区二区三区| 自拍日韩欧美| 日韩av一区二区三区| 成人片免费看| 亚洲精品欧洲| 亚洲欧洲美洲av| 日韩视频一二区| 超碰在线99| 亚洲人成网77777色在线播放| 久久精品人人| 久久最新视频| 鲁鲁在线中文| 日日夜夜免费精品视频| 成人片免费看| 久久精品凹凸全集| 一区二区小说| 久久av电影| 一区免费视频| 美女国产一区二区三区| 蜜臀久久99精品久久久久宅男 | 婷婷成人基地| 国产经典一区| 自拍自偷一区二区三区| 成人片免费看|