IDEA plug-ins commonly used by programmers: github.com/silently952…
Fully open source Taoke project: github.com/silently952…
Wechat official account: Beta learning Java
preface
If you design an event system with many events, each event has a different weight value, and the system needs to handle the events with a higher weight first, you need to use priority queues. In this article, we will learn the common ways to implement priority queues
Queue API Definition
Before implementing it, we first need to define the API of priority queue. Priority queue is an abstract data structure that we can still modify based on the queue API we used previously. To see the previous implementation of the queue, check out “Interview season, Are you sure you want to review data structures?”
public interface Queue<T> extends Iterable<T> { void enqueue(T item); T dequeue(); Int size(); boolean isEmpty(); }Copy the code
Enqueue and deQueue are the main methods we need to implement, and they are the core methods of priority queues
A preliminary version of the implementation
Abstract class of the queue API
public abstract class AbstractQueue<T> implements Queue<T> { private Comparator<T> comparator; public AbstractQueue(Comparator<T> comparator) { this.comparator = comparator; } public boolean less(T a, T b) { return comparator.compare(a, b) < 0; } public void exch(T[] array, int i, int j) { T tmp = array[i]; array[i] = array[j]; array[j] = tmp; }}Copy the code
Based on unordered array implementation
The simplest implementation of a priority queue can refer to the implementation of stack in “Interview season, are you sure to review the data structure”. Enqueue and stack push are implemented in the same way. Dequeue can refer to the implementation of selective sorting, loop through the array, find the maximum value and the last element of the array swap. Then delete it;
public class DisorderPriorityQueue<T> extends AbstractQueue<T> { private T[] queue; private int size; public DisorderPriorityQueue(int max, Comparator<T> comparator) { super(comparator); this.queue = (T[]) new Object[max]; } @Override public void enqueue(T item) { queue[size++] = item; } @Override public T dequeue() { int index = 0; for (int i = 1; i < size; i++) { if (less(queue[index], queue[i])) { index = i; } } size--; exch(queue, index, size); T data = queue[size]; queue[size] = null; return data; } // omit other functions}Copy the code
Here, only a fixed priority queue is implemented, how to achieve automatic capacity expansion? Also check out this article, “Interview season, are you sure I’m not going to review data structures?” The time complexity of enqueue based on unordered array is O(1), the time complexity of dequeue is O(n).
Implementation based on ordered array
The implementation based on ordered array is to ensure that the array is in order when entering the queue, so the maximum value can be directly deleted when exiting the queue; The procedure of insert is similar to the operation of insert sort
public class OrderPriorityQueue<T> extends AbstractQueue<T> { private T[] queue; private int size; public OrderPriorityQueue(int max, Comparator<T> comparator) { super(comparator); this.queue = (T[]) new Object[max]; } @Override public void enqueue(T item) { queue[size++] = item; for (int i = size - 1; i > 1 && less(queue[i], queue[i - 1]); i--) { exch(queue, i, i - 1); } } @Override public T dequeue() { size--; T data = queue[size]; queue[size] = null; return data; } // omit other functions}Copy the code
Enqueue time is O(n), dequeue time is O(1)
Based on linked list implementation
The linked list-based implementation is similar to the above, and you can implement it yourself if you are interested
Is confirmed in the interview of the season, brother? “to review the data structure in our implementation of the operation of the stack and queue can be completed within the time constant, but the realization of the priority queue from the above process, we found that the realization of the primary version insert or delete a maximum operating in the worst case would be linear time.
Binary heap implementation
Definition of binary heap
In a binary heap, each node will be greater than or equal to its children, also known as heap order; The root node is the largest node.
Binary heap representation:
Important: in a binary heap, the parent node of position K is at k/2, and its two children are at 2k and 2K +1; Based on this, we can represent the binary heap as an array and find the parent and child nodes by moving the subscripts of the array
As elements are inserted and deleted, they break the heap order, so we need to do something to keep the heap in order again. There are two main cases: when a node’s priority rises, we need to restore heap order from the bottom up (sink); When a node’s priority drops, we need to restore heap order from the top down (float up)
Restore heap order from top to bottom (float up)
private void swim(int k) { while (k > 0 && less(queue[k / 2], queue[k])) { exch(queue, k / 2, k); k = k / 2; }}Copy the code
Find the location k/2 of the parent node according to the current node K, compare the current node and the parent node, if the parent node is larger than the parent node, switch until find a parent node larger than the current node or has floated to the root node
Restore heap order from bottom up (sink)
private void sink(int k) { while (2 * k <= size) { int i = 2 * k; if (less(queue[i], queue[i + 1])) { i++; } if (less(queue[i], queue[k])) { break; } exch(queue, i, k); k = i; }}Copy the code
Binary heap implements priority queues
- Queue operations: Add new elements to the end of the array, float the new elements up to the appropriate position, and increase the heap size
- Queue operation: remove the largest root node, then put the last element at the top, lower the top element in place, reduce the heap size
public class BinaryHeapPriorityQueue<T> extends AbstractQueue<T> { private T[] queue; private int size; public BinaryHeapPriorityQueue(int max, Comparator<T> comparator) { super(comparator); this.queue = (T[]) new Object[max + 1]; } @Override public void enqueue(T item) { this.queue[++size] = item; this.swim(size); } @Override public T dequeue() { T max = this.queue[1]; exch(this.queue, 1, size--); this.queue[size + 1] = null; This.sink (1); return max; } // omit other functions}Copy the code
Note:
The first position in the array is not used because it is easy to calculate the index positions of the parent and child nodes. If you use the first position, how do you calculate the position of the child node and the parent node?
The heap-based implementation, which has complex enqueue and enqueue times for both pairs, is logN, addressing the problems of the initial implementation.
Array size dynamic expansion and reduction can still refer to the implementation of the previous stack
All source code has been put into github repository github.com/silently952…
Finally (pay attention, don’t get lost)
There may be more or less deficiencies and mistakes in the article, suggestions or opinions are welcome to comment and exchange.
Finally, writing is not easy, please do not white whoring I yo, I hope friends can point comment attention to three even, because these are all the power source I share 🙏