Files
Mindustry/core/src/mindustry/ai/PathfindQueue.java
2022-02-09 17:13:02 -05:00

146 lines
5.0 KiB
Java

package mindustry.ai;
import arc.util.*;
/** A priority queue. */
@SuppressWarnings("unchecked")
public class PathfindQueue{
private static final double CAPACITY_RATIO_LOW = 1.5f;
private static final double CAPACITY_RATIO_HI = 2f;
/**
* Priority queue represented as a balanced binary heap: the two children of queue[n] are queue[2*n+1] and queue[2*(n+1)]. The
* priority queue is ordered by the elements' natural ordering: For each node n in the heap and each descendant d of n, n <= d.
* The element with the lowest value is in queue[0], assuming the queue is nonempty.
*/
public int[] queue;
/** Weights of each object in the queue. */
public float[] weights;
/** The number of elements in the priority queue. */
public int size = 0;
public PathfindQueue(){
this(12);
}
public PathfindQueue(int initialCapacity){
this.queue = new int[initialCapacity];
this.weights = new float[initialCapacity];
}
public boolean empty(){
return size == 0;
}
/**
* Inserts the specified element into this priority queue. If {@code uniqueness} is enabled and this priority queue already
* contains the element, the call leaves the queue unchanged and returns false.
* @return true if the element was added to this queue, else false
* @throws ClassCastException if the specified element cannot be compared with elements currently in this priority queue
* according to the priority queue's ordering
* @throws IllegalArgumentException if the specified element is null
*/
public boolean add(int e, float weight){
int i = size;
if(i >= queue.length) growToSize(i + 1);
size = i + 1;
if(i == 0){
queue[0] = e;
weights[0] = weight;
}else{
siftUp(i, e, weight);
}
return true;
}
/**
* Retrieves, but does not remove, the head of this queue. If this queue is empty, {@code 0} is returned.
* @return the head of this queue
*/
public int peek(){
return size == 0 ? 0 : queue[0];
}
/** Removes all of the elements from this priority queue. The queue will be empty after this call returns. */
public void clear(){
size = 0;
}
/**
* Retrieves and removes the head of this queue, or returns {@code null} if this queue is empty.
* @return the head of this queue, or {@code null} if this queue is empty.
*/
public int poll(){
if(size == 0) return 0;
int s = --size;
int result = queue[0];
int x = queue[s];
if(s != 0) siftDown(0, x, weights[s]);
return result;
}
/**
* Inserts item x at position k, maintaining heap invariant by promoting x up the tree until it is greater than or equal to its
* parent, or is the root.
* @param k the position to fill
* @param x the item to insert
*/
private void siftUp(int k, int x, float weight){
while(k > 0){
int parent = (k - 1) >>> 1;
int e = queue[parent];
if(weight >= weights[parent]) break;
queue[k] = e;
weights[k] = weights[parent];
k = parent;
}
queue[k] = x;
weights[k] = weight;
}
/**
* Inserts item x at position k, maintaining heap invariant by demoting x down the tree repeatedly until it is less than or
* equal to its children or is a leaf.
* @param k the position to fill
* @param x the item to insert
*/
private void siftDown(int k, int x, float weight){
int half = size >>> 1; // loop while a non-leaf
while(k < half){
int child = (k << 1) + 1; // assume left child is least
int c = queue[child];
int right = child + 1;
if(right < size && weights[child] > weights[right]){
c = queue[child = right];
}
if(weight <= weights[child]) break;
queue[k] = c;
weights[k] = weights[child];
k = child;
}
queue[k] = x;
weights[k] = weight;
}
/**
* Increases the capacity of the array.
* @param minCapacity the desired minimum capacity
*/
private void growToSize(int minCapacity){
if(minCapacity < 0) // overflow
throw new ArcRuntimeException("Capacity upper limit exceeded.");
int oldCapacity = queue.length;
// Double size if small; else grow by 50%
int newCapacity = (int)((oldCapacity < 64) ? ((oldCapacity + 1) * CAPACITY_RATIO_HI) : (oldCapacity * CAPACITY_RATIO_LOW));
if(newCapacity < 0) // overflow
newCapacity = Integer.MAX_VALUE;
if(newCapacity < minCapacity) newCapacity = minCapacity;
int[] newQueue = new int[newCapacity];
float[] newWeights = new float[newCapacity];
System.arraycopy(queue, 0, newQueue, 0, size);
System.arraycopy(weights, 0, newWeights, 0, size);
queue = newQueue;
weights = newWeights;
}
}