亚洲激情专区-91九色丨porny丨老师-久久久久久久女国产乱让韩-国产精品午夜小视频观看

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

priorityqueue怎么在JAVA中使用

發布時間:2021-01-21 15:06:47 來源:億速云 閱讀:148 作者:Leah 欄目:開發技術

本篇文章為大家展示了priorityqueue怎么在JAVA中使用,內容簡明扼要并且容易理解,絕對能使你眼前一亮,通過這篇文章的詳細介紹希望你能有所收獲。

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

總體介紹

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

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

priorityqueue怎么在JAVA中使用

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

leftNo = parentNo*2+1

rightNo = parentNo*2+2

parentNo = (nodeNo-1)/2

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

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

方法剖析

add()和offer()

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

priorityqueue怎么在JAVA中使用

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

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

priorityqueue怎么在JAVA中使用

代碼也就非常簡潔:

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

remove()和poll()

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

priorityqueue怎么在JAVA中使用

代碼如下:

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

priorityqueue怎么在JAVA中使用

具體代碼如下:

//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;
}

上述內容就是priorityqueue怎么在JAVA中使用,你們學到知識或技能了嗎?如果還想學到更多技能或者豐富自己的知識儲備,歡迎關注億速云行業資訊頻道。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

青龙| 双峰县| 姜堰市| 蕉岭县| 台前县| 新密市| 卢氏县| 大余县| 绥中县| 富民县| 鹤壁市| 延庆县| 济宁市| 南通市| 陆良县| 巴里| 安化县| 万全县| 随州市| 浪卡子县| 屯门区| 固镇县| 长泰县| 项城市| 山东省| 宜兴市| 盘锦市| 高清| 稻城县| 辰溪县| 准格尔旗| 寿阳县| 九龙坡区| 武隆县| 黄骅市| 拜城县| 建宁县| 定西市| 衢州市| 清涧县| 柘城县|