亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > Java > 正文

深入理解Java PriorityQueue

2019-11-06 07:03:46
字體:
來源:轉載
供稿:網友

引用自此博客文章,感謝CarpenterLee

深入理解java PRiorityQueue

本文github地址

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

總體介紹

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

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

優先隊列的數組表示的小頂堆實現

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

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

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

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

方法剖析

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

offer()并調整

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

//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下標處的那個元素即可。

peek()

代碼也就非常簡潔:

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

remove()poll()

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

poll()并調整 代碼如下:

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()即可。此處不再贅述。

remove(Object o)

具體代碼如下:

//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;}
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
久久免费视频这里只有精品| 亚洲午夜精品久久久久久性色| 欧美xxxx做受欧美.88| 国产日韩综合一区二区性色av| 国内精品久久久久久久久| 国产欧美日韩免费看aⅴ视频| 欧美日韩xxxxx| 91久久国产精品91久久性色| 久久艹在线视频| 国产精品激情av在线播放| 成人免费看片视频| 日韩一区二区久久久| 亚洲影影院av| 精品美女久久久久久免费| 亚洲福利视频久久| 久久精品成人欧美大片| 欧美成人中文字幕| 久久视频精品在线| 日韩久久午夜影院| 欧美日韩xxxxx| 亚州av一区二区| 欧美成人高清视频| 国产精品综合网站| 国内精品久久久久久影视8| 97国产精品视频人人做人人爱| 国产精品高清网站| 日本精品一区二区三区在线| 亚洲a级在线播放观看| 日本高清久久天堂| 欧美大人香蕉在线| 欧美精品久久一区二区| 欧美精品久久久久久久免费观看| 国产日韩欧美91| 精品一区精品二区| 欧美限制级电影在线观看| 在线视频欧美日韩精品| 伊人久久免费视频| 亚洲欧美制服中文字幕| 日韩在线免费视频观看| 日韩欧美中文字幕在线观看| 国产精品极品美女粉嫩高清在线| 精品亚洲一区二区三区在线观看| 欧美成人精品三级在线观看| 成人免费视频97| 亚洲国产精品va在线看黑人| 国产精品直播网红| 亚洲精品成人网| 主播福利视频一区| 欧美激情影音先锋| 91中文字幕在线观看| 国产精品久久久久久久久| 欧美精品电影免费在线观看| 国产精品视频26uuu| 精品久久久久久中文字幕| 国产成人综合一区二区三区| 夜夜嗨av色综合久久久综合网| 日韩电影中文字幕| 国模私拍一区二区三区| 久久久女人电视剧免费播放下载| 亚洲一区二区三区视频| 国产香蕉精品视频一区二区三区| 7m第一福利500精品视频| 成人在线播放av| 国产啪精品视频网站| 4444欧美成人kkkk| 国产成人综合久久| 91在线无精精品一区二区| 成人性教育视频在线观看| 亚洲色图五月天| 福利一区视频在线观看| 精品久久久久久久久久国产| 欧美黑人国产人伦爽爽爽| 欧美性猛交xxxx乱大交| 国产精品免费久久久| 亚洲男人第一av网站| 亚洲2020天天堂在线观看| 国产午夜精品全部视频播放| 蜜月aⅴ免费一区二区三区| 精品偷拍各种wc美女嘘嘘| 热草久综合在线| 国产午夜精品一区理论片飘花| 国产精品啪视频| 亚洲成色777777在线观看影院| 亚洲一区二区福利| 国内精品小视频| 国产精品久久久久久久久久东京| 国产一区二区三区在线播放免费观看| 国产精品免费久久久久久| 国产精品久久久久77777| 亚洲最新视频在线| 久久久爽爽爽美女图片| 国产精品视频中文字幕91| 九九热这里只有精品6| 亚洲午夜未删减在线观看| 国产精品一区av| 国产精品日韩精品| 国产精品99久久久久久久久久久久| 欧美在线一区二区三区四| 精品国产乱码久久久久久虫虫漫画| 亚洲欧美综合精品久久成人| 自拍偷拍亚洲精品| 欧美激情一区二区三区久久久| 91精品国产网站| 国产精品扒开腿做爽爽爽的视频| 91欧美精品成人综合在线观看| 91av视频在线免费观看| 欧美中在线观看| 欧美成在线视频| 色偷偷91综合久久噜噜| 国外成人在线视频| 在线播放国产精品| 国产综合久久久久久| 成人综合网网址| 国产精品久久久久久久久久ktv| 欧美日韩国产成人高清视频| 中文字幕日韩在线视频| 国产免费一区二区三区在线能观看| 羞羞色国产精品| 久久久噜噜噜久久中文字免| 日韩美女在线观看| 国产精品无码专区在线观看| 欧美三级欧美成人高清www| 97色在线视频| 日韩av在线精品| 久99久在线视频| 日韩欧美在线观看| 亚洲女同性videos| 91在线免费网站| 日韩小视频网址| 琪琪亚洲精品午夜在线| 久久久久久国产三级电影| 国产欧美最新羞羞视频在线观看| 日韩电影中文字幕| 日韩中文字幕不卡视频| 成人免费看片视频| 国产精品久久久久久久电影| 影音先锋欧美精品| 亚洲午夜女主播在线直播| 欧美xxxx做受欧美.88| 精品国产区一区二区三区在线观看| 欧美午夜影院在线视频| 精品国产拍在线观看| 欧美日韩在线另类| 精品国产31久久久久久| 亚洲欧洲国产伦综合| 久久久精品日本| 浅井舞香一区二区| 永久免费看mv网站入口亚洲| 日韩av中文在线| 亚洲欧美日韩一区二区三区在线| 亚洲天堂男人天堂女人天堂| 上原亚衣av一区二区三区| 欧美激情成人在线视频| 日韩性xxxx爱| 欧美亚洲日本网站| 欧美与欧洲交xxxx免费观看| 国产精品一区二区三| 日本一欧美一欧美一亚洲视频| 不卡在线观看电视剧完整版| 亚洲精品中文字幕av| 九九热这里只有在线精品视| 91精品国产免费久久久久久| 91热精品视频| 欧美高清无遮挡|