美文网首页
什么是优先队列?

什么是优先队列?

作者: 西三旗靓仔 | 来源:发表于2020-02-17 20:39 被阅读0次

    优先队列是二叉堆的一个应用,普通队列的特点是先进先出(FIFO)。优先队列不再遵循先入先出的原则,而是分为两种情况:

    最大优先队列,无论入队顺序,当前最大的元素优先出队。

    最小优先队列,无论入队顺序,当前最小的元素优先出队。

    比如有一个最大优先队列,它的最大元素是8,那么虽然元素8并不是队首元素,但出队的时候仍然让元素8首先出队:

    什么是优先队列?

    要满足以上需求,利用线性数据结构并非不能实现,但是时间复杂度较高,最坏时间复杂度O(n),并不是最理想的方式。而二叉堆却很适合,二叉堆的特性:

    1.最大堆的堆顶是整个堆中的最大元素

    2.最小堆的堆顶是整个堆中的最小元素**

    因此,我们可以用最大堆来实现最大优先队列,每一次入队操作就是堆的插入操作,每一次出队操作就是堆顶删除操作。因而优先队列的入队和出队的时间复杂度均为log(n)

    入队操作

    1.插入新节点5

    什么是优先队列?

    2.新节点5上浮到合适位置。

    什么是优先队列?

    出队操作

    1.把原堆顶节点10“出队”

    什么是优先队列?

    2.最后一个节点1替换到堆顶位置

    什么是优先队列?

    3.节点1下沉,节点9成为新堆顶

    什么是优先队列?

    代码实现

    public class TestMain {
      public static void main(String[] args) {
        int[] a = new int[20];
        for (int i = 0; i < a.length; i++) {
          int temp = (int)(StdRandom.random()*100);
          a[i] = temp;
        }
        for (int i : a) {
          System.out.print(i+" ");
        }
        System.out.println();
        PQHeap pq = new PQHeap();
        for (int i = 0; i < 20; i++) {
          pq.insert(a[i]);
        }
        System.out.println();
        for (int i = 0; i < 20; i++) {
          System.out.print(pq.delMax()+" ");
        }
      }
    }
    /*
     * 优先队列的堆实现
     * 二叉堆,每个元素有两个子元素,两个子元素均比自己小
     */
    class PQHeap{
      private int[] a;
      private int p = 1;
      public PQHeap() {
        a = new int[2];
      }
      /*
       * 插入元素
       */
      public void insert(int elements) {
        if (p == a.length) { resize(a.length*2); }
        a[p++] = elements;
        swim(p - 1); //将刚插入的元素上浮到正确位置
      }
      /*
       * 返回并删除最大元素
       */
      public int delMax() {
        if (p == a.length/4) { resize(a.length/2); }
        int result = a[1]; //找出最大元素
        exch(1, --p); //将最后一个元素移到顶端
        a[p] = 0; 
        sink(1); //将刚移上去的元素沉下去,使堆有序
        return result;
      }
      public boolean isEmpty() {
        return p == 0;
      }
      public int size() {
        return p;
      }
      public void show() {
        for (int i : a) {
          System.out.print(i+" ");
        }
        System.out.println();
      }
      /*
       * 上浮元素
       */
      private void swim(int k) { //将元素与父元素比较,比父元素大则换位置
        while (k > 1 && a[k/2] < a[k]) { 
          exch(k/2, k);
          k = k/2;
        }
      }
      private void sink(int k) { //将元素与子元素比较,比子元素小则和两个中较大的那个换位置
        while (2*k < p && (a[k] < a[2*k + 1]) || (a[k] < a[2*k])) {
          if (a[2*k + 1] > a[2*k]) { exch(k, 2*k + 1); k = 2*k + 1; }
          else           { exch(k, 2*k); k = 2*k; }
        }
      }
      private void resize(int length) {
        int[] b = new int[length]; //将数组长度改变
        for (int i = 0; i < p; i++) { //将数组复制
          b[i] = a[i];
        }
        a = b; //让a指向b的内存空间
      }
      /*
       * 交换
       */
      private void exch (int i, int j) {
        int t = a[i];
        a[i] = a[j];
        a[j] = t;
      }
    }
    

    相关文章

      网友评论

          本文标题:什么是优先队列?

          本文链接:https://www.haomeiwen.com/subject/ntpifhtx.html