use std::{cmp::Ordering, collections::BinaryHeap}; use crate::PriorityQueue; pub(crate) struct Item { key: K, value: T, } impl std::fmt::Debug for Item { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { write!(f, "({:?}: {:?})", self.key, self.value) } } impl std::fmt::Debug for PriorityQueue { fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { std::fmt::Debug::fmt(&self.0, f) } } impl PartialEq for Item { fn eq(&self, other: &Self) -> bool { Ord::cmp(self, other).is_eq() } } impl Eq for Item {} impl PartialOrd for Item { fn partial_cmp(&self, other: &Self) -> Option { Ord::cmp(self, other).into() } } impl Ord for Item { fn cmp(&self, other: &Self) -> Ordering { self.key.cmp(&other.key) } } impl Clone for Item where K: Clone, T: Clone, { fn clone(&self) -> Self { Item { key: self.key.clone(), value: self.value.clone(), } } } impl Clone for PriorityQueue where K: Clone + Ord, T: Clone, { fn clone(&self) -> Self { Self(self.0.clone()) } } impl PriorityQueue { /// Create a new instance. pub fn new() -> Self { PriorityQueue(Default::default()) } /// Insert `value` so that it is ordered according to `key`. pub fn insert(&mut self, key: K, value: T) { self.0.push(Item { key, value }); } /// Pop the highest-priority item value off the queue. pub fn pop_value(&mut self) -> Option { self.0.pop().map(|t| t.value) } /// Pop the highest-priority item key and value off the queue. pub fn pop(&mut self) -> Option<(K, T)> { self.0.pop().map(|t| (t.key, t.value)) } /// Iterate all items ordered from highest to lowest priority. pub fn iter_unordered(&self) -> impl Iterator { self.0.iter().map(|t| &t.value) } /// Turn this instance into an iterator over its keys and values in arbitrary order. pub fn into_iter_unordered(self) -> impl Iterator { self.0.into_vec().into_iter().map(|item| (item.key, item.value)) } /// Return true if the queue is empty. pub fn is_empty(&self) -> bool { self.0.is_empty() } /// Returns the greatest item `(K, T)` tuple, as ordered by `K`, if the queue is not empty, without removing it. pub fn peek(&self) -> Option<(&K, &T)> { self.0.peek().map(|e| (&e.key, &e.value)) } /// Drop all items from the queue, without changing its capacity. pub fn clear(&mut self) { self.0.clear() } } impl FromIterator<(K, T)> for PriorityQueue { fn from_iter>(iter: I) -> Self { let mut q = PriorityQueue(BinaryHeap::new()); for (k, v) in iter { q.insert(k, v); } q } }