Refactor into different crates

This commit is contained in:
hal8174 2025-01-18 17:30:55 +01:00
parent 94473c64e0
commit dfdeae5638
82 changed files with 624 additions and 647 deletions

View file

@ -0,0 +1,232 @@
use std::fmt::Debug;
pub mod fibonacci_heap;
pub trait PriorityQueue<Item>
where
Item: PartialOrd,
{
type Handle;
fn new() -> Self;
// fn with_capacity() -> Self {
// Self::new()
// }
fn insert(&mut self, item: Item) -> Self::Handle;
fn pop_min(&mut self) -> Option<Item>;
fn decrease_key(&mut self, handle: &Self::Handle, f: impl Fn(&mut Item));
}
#[derive(Debug)]
pub struct BinaryHeap<Item> {
nextfree: usize,
data: Vec<BinaryHeapEntry<Item>>,
}
#[derive(Debug)]
struct BinaryHeapEntry<Item> {
id: usize,
item: Item,
}
impl<Item> BinaryHeap<Item>
where
Item: PartialOrd,
{
fn downheap(&mut self, index: usize) {
let left = 2 * index + 1;
let right = 2 * index + 2;
if right < self.data.len() {
let smaller = if self.data[left].item < self.data[right].item {
left
} else {
right
};
if self.data[index].item > self.data[smaller].item {
self.data.swap(index, smaller);
self.downheap(smaller);
}
} else if left < self.data.len() && self.data[index].item > self.data[left].item {
self.data.swap(index, left);
self.downheap(left);
}
}
fn upheap(&mut self, index: usize) {
if index > 0 {
let parent = (index - 1) / 2;
if self.data[parent].item > self.data[index].item {
self.data.swap(parent, index);
self.upheap(parent);
}
}
}
fn search(&self, id: usize) -> Option<usize> {
for (i, d) in self.data.iter().enumerate() {
if d.id == id {
return Some(i);
}
}
None
}
}
impl<Item> PriorityQueue<Item> for BinaryHeap<Item>
where
Item: PartialOrd + Clone,
{
type Handle = usize;
fn insert(&mut self, item: Item) -> Self::Handle {
self.data.push(BinaryHeapEntry {
id: self.nextfree,
item: item.clone(),
});
self.upheap(self.data.len() - 1);
self.nextfree += 1;
self.nextfree - 1
}
fn pop_min(&mut self) -> Option<Item> {
if self.data.is_empty() {
None
} else {
let d = self.data.swap_remove(0);
self.downheap(0);
Some(d.item)
}
}
fn decrease_key(&mut self, handle: &Self::Handle, f: impl Fn(&mut Item)) {
if let Some(index) = self.search(*handle) {
f(&mut self.data[index].item);
self.upheap(index);
}
}
fn new() -> Self {
Self {
data: Vec::new(),
nextfree: 0,
}
}
}
#[derive(Debug)]
pub struct Trace<P> {
inner: P,
}
impl<P, I> PriorityQueue<I> for Trace<P>
where
I: PartialOrd + Clone + Debug,
P: PriorityQueue<I>,
{
type Handle = P::Handle;
fn new() -> Self {
println!("New priority queue.");
Self { inner: P::new() }
}
fn insert(&mut self, item: I) -> Self::Handle {
println!("Insert: {item:?}");
self.inner.insert(item)
}
fn pop_min(&mut self) -> Option<I> {
let min = self.inner.pop_min();
println!("Pop min: {min:?}");
min
}
fn decrease_key(&mut self, handle: &Self::Handle, f: impl Fn(&mut I)) {
self.inner.decrease_key(handle, |i| {
let old_i = i.clone();
f(i);
println!("Decrease key: {old_i:?} -> {i:?}");
})
}
}
#[cfg(test)]
mod tests {
use super::fibonacci_heap::FibonacciHeap;
use super::BinaryHeap;
use super::PriorityQueue;
macro_rules! test_generics {
($($fun_mul:ident )+ ; $gen:ident $($gen_mul:ident)+) => {
test_generics!(@internal_mod $($fun_mul )*; $gen);
test_generics!($($fun_mul )*; $($gen_mul )*);
};
($($fun_mul:ident )+ ;$gen:ident) => {
test_generics!(@internal_mod $($fun_mul )*; $gen);
};
(@internal_mod $($fun_mul:ident )+; $gen:ident) => {
#[allow(non_snake_case)]
mod $gen {
test_generics!(@internal $($fun_mul )*; $gen);
}
};
(@internal $fun:ident $($fun_mul:ident )+; $gen:ident) => {
test_generics!(@internal $fun; $gen);
test_generics!(@internal $($fun_mul )*; $gen);
};
(@internal $fun:ident; $gen:ident) => {
#[test]
fn $fun() {
super::super::$fun::<super::super::$gen<usize>>();
}
}
}
mod test_macro {
test_generics!(basic_generic decrease_key_generic; BinaryHeap FibonacciHeap);
}
fn basic_generic<T: PriorityQueue<usize>>() {
let mut q = T::new();
q.insert(2);
q.insert(3);
q.insert(10);
q.insert(12);
q.insert(9);
q.insert(6);
q.insert(4);
q.insert(5);
q.insert(8);
q.insert(7);
q.insert(11);
q.insert(1);
assert_eq!(q.pop_min(), Some(1));
assert_eq!(q.pop_min(), Some(2));
assert_eq!(q.pop_min(), Some(3));
assert_eq!(q.pop_min(), Some(4));
assert_eq!(q.pop_min(), Some(5));
assert_eq!(q.pop_min(), Some(6));
assert_eq!(q.pop_min(), Some(7));
assert_eq!(q.pop_min(), Some(8));
assert_eq!(q.pop_min(), Some(9));
assert_eq!(q.pop_min(), Some(10));
assert_eq!(q.pop_min(), Some(11));
assert_eq!(q.pop_min(), Some(12));
assert_eq!(q.pop_min(), None);
}
fn decrease_key_generic<T: PriorityQueue<usize>>() {
let mut q = T::new();
q.insert(4);
let h = q.insert(5);
q.decrease_key(&h, |i| *i -= 3);
assert_eq!(q.pop_min(), Some(2));
assert_eq!(q.pop_min(), Some(4));
}
}