一、 索引优先队列

  上一章节的数据结构(八):优先队列-大小优先中,能够快速的从队列中取出最大最小值并删除,但有个缺点,就是无法通过索引快速的找到某个值,并且修改它,对于能快速根据索引查找到值的需求

  我们使用索引优先队列来实现。

二、 最小索引优先队列思路

  步骤一:实现索引优先队列,直观的想法是对数据元素添加对应的key,方便通过key找到value,由于优先队列我们使用数组实现,因此可以使用索引-值的方式来表示,如下图:

  items[]:{ "S", "O", "R", "T", "E", "X", "A", "M", "P", "L", "E" }

   

  这样我们便能通过items[i]获取到值

  步骤二:

  步骤一将元素存储至堆数组后,发现不符合堆有序,此时若对items数组进行堆有序调整,则会破坏元素在items的下标,再通过items[i]获取时会有问题,此时我们引入一个辅助数组pq[]

  pq的存放元素在 items中的索引,对堆进行有序处理时,只需处理pq[i]即可:

  有序后的堆:

   

   

  此时的pq数组即是一个有序堆

  步骤三:

  通过步骤二的调整后,我们对堆有序的调整,只需通过pq进行调整,items元素保持不变,当我们做出items[0]=J的操作后,只需通过遍历pq数组,对pq[9]做上浮和下沉调整即可,保证pq堆有序

  但在数据比较大的场景下,上述的遍历找到pq[i]的成本就变得很高,此时我们引入另一个辅助数组qp[],qp的存在是为了更快找到pq中值对应的items的索引,此时我们可以对pq做一个倒转

  及qp的索引是pq的值,qp的值是pq的索引,见下图:

   

  有了qp数组后,当我们做出items[0]=J的操作后,只需通过qp[0]获取到9,再去对pq[9]做上浮和下沉处理,及可在修改队列元素的同时,又保证堆有序。

三、 代码实现

  如下代码基于如上三个步骤实现了索引最小优先队列:

/**

 * 索引最小优先队列

 *

 * @author jiyukai

 * @param <T>

 */

public class IndexMinPriorityQueue<T extends Comparable<T>> {

       // 存储具体元素的堆

       private T[] items;

       // 保存每个元素在items中的索引,pq需要堆有序

       private int[] pq;

       // 保存pq数组索引和值的逆序

       private int[] qp;

       // 堆中元素的个数

       private int N;

       /**

        * 初始化索引优先队列的大小

        *

        * @param size

        */

       public IndexMinPriorityQueue(int size) {

              items = (T[]) new Comparable[size + 1];

              N = 0;

              pq = new int[size + 1];

              qp = new int[size + 1];

              for (int i = 0; i < qp.length; i++) {

                     qp[i] = -1;

              }

       }

       /**

        * 优先队列大小

        *

        * @return

        */

       public int size() {

              return N;

       }

       /**

        * 是否为空

        *

        * @return

        */

       public boolean isEmpty() {

              return N == 0;

       }

       /**

        * 元素大小比较

        *

        * @param i

        * @param j

        * @return

        */

       public boolean bigger(int i, int j) {

              return items[i].compareTo(items[j]) > 0;

       }

       /**

        * 位置交换

        *

        * @param i

        * @param j

        */

       private void switchPos(int i, int j) {

              int temp = pq[i];

              pq[i] = pq[j];

              pq[j] = temp;

              // 修改qp

              qp[pq[i]] = i;

              qp[pq[j]] = j;

       }

       /**

        * 判断k位置对应的元素是否存在

        *

        * @param k

        * @return

        */

       public boolean contains(int k) {

              return qp[k] != -1;

       }

       /**

        * 最小元素的索引

        *

        * @return

        */

       public int minIndex() {

              return pq[1];

       }

       /**

        * i位置处插入元素t

        *

        * @param i

        * @param t

        * @throws Exception

        */

       public void insert(int i, T t) throws Exception {

              // 判断位置i处是否存在元素t

              if (contains(i)) {

                     throw new Exception(i + "位置处已有元素" + t);

              }

              // item的i处插入t

              items[i] = t;

              // 元素个数+1

              N++;

              // 插入到最后一个元素,因此pq的N位置处为item的i

              pq[N] = i;

              // 修改qp的i位置处为N

              qp[i] = N;

              // 上浮处理N,让堆有序

              swim(N);

       }

       /**

        * 删除最小的元素并返回这个元素关联的索引

        *

        * @return

        */

       public int delMin() {

              // 获取items中最小的元素的索引

              int min = pq[1];

              // 交换最小元素和N位置处的元素

              switchPos(1, N);

              // 删除qp

              qp[pq[N]] = -1;

              // 删除pq

              pq[N] = -1;

              // N位置处的元素置空,即删除掉最小元素

              items[min] = null;

              // 元素个数减1

              N--;

              // 对1位置处的元素进行下沉处理

              sink(1);

              return min;

       }

       /**

        * 删除索引i关联的元素

        *

        * @param i

        */

       public void delete(int i) {

              // 获取items i位置的元素

              int index = pq[i];

              // 交换最小元素和N位置处的元素

              switchPos(index, N);

              // 删除qp

              qp[pq[N]] = -1;

              // 删除pq

              pq[N] = -1;

              // N位置处的元素置空,即删除掉最小元素

              items[i] = null;

              // 元素个数减1

              N--;

              // 对i位置处的元素进行下沉处理

              sink(index);

             

              // 对i位置处的元素进行上浮处理

              swim(index);

       }

       /**

        * 把索引i处的元素修改为t

        *

        * @param i

        * @param t

        */

       public void changeItem(int i, T t) {

              //修改items中i位置处的元素为t

              items[i] = t;

              //获取i位置处元素在pq中的索引

              int k = pq[i];

              //上浮

              swim(k);

              //下沉

              sink(k);

       }

       /**

        * 上浮k位置处的元素

        *

        * @param k

        */

       private void swim(int k) {

              while(k>1) {

                     if(bigger(k, k/2)) {

                            switchPos(k, k/2);

                     }

                    

                     k = k/2;

              }

       }

       /**

        * 下沉k位置处的元素

        *

        * @param k

        */

       private void sink(int k) {

              while(2 * k <= N) {

                     int min = 2 * k;

                     if(bigger(2*k, 2*k+1)) {

                            min = 2*k+1;

                     }

                    

                     if(bigger(min,k)) {

                            break;

                     }

                    

                     switchPos(min, k);

                    

                     k = min;

              }

       }

}


评论关闭
IT源码网

微信公众号号:IT虾米 (左侧二维码扫一扫)欢迎添加!