Skip to content

Commit

Permalink
Implement Combinations::nth
Browse files Browse the repository at this point in the history
  • Loading branch information
kinto-b authored and Philippe-Cholet committed Apr 15, 2024
1 parent 5be2e8d commit 4cbb6d3
Showing 1 changed file with 72 additions and 29 deletions.
101 changes: 72 additions & 29 deletions src/combinations.rs
Original file line number Diff line number Diff line change
Expand Up @@ -14,22 +14,23 @@ pub struct Combinations<I: Iterator> {
indices: Vec<usize>,
pool: LazyBuffer<I>,
first: bool,
done: bool,
}

impl<I> Clone for Combinations<I>
where
I: Clone + Iterator,
I::Item: Clone,
{
clone_fields!(indices, pool, first);
clone_fields!(indices, pool, first, done);
}

impl<I> fmt::Debug for Combinations<I>
where
I: Iterator + fmt::Debug,
I::Item: fmt::Debug,
{
debug_fmt_fields!(Combinations, indices, pool, first);
debug_fmt_fields!(Combinations, indices, pool, first, done);
}

/// Create a new `Combinations` from a clonable iterator.
Expand All @@ -41,6 +42,7 @@ where
indices: (0..k).collect(),
pool: LazyBuffer::new(iter),
first: true,
done: false,
}
}

Expand Down Expand Up @@ -70,6 +72,7 @@ impl<I: Iterator> Combinations<I> {
/// elements.
pub(crate) fn reset(&mut self, k: usize) {
self.first = true;
self.done = false;

if k < self.indices.len() {
self.indices.truncate(k);
Expand All @@ -90,10 +93,56 @@ impl<I: Iterator> Combinations<I> {
indices,
pool,
first,
done: _,
} = self;
let n = pool.count();
(n, remaining_for(n, first, &indices).unwrap())
}

/// Initialises the iterator by filling a buffer with elements from the
/// iterator.
fn init(&mut self) {
self.pool.prefill(self.k());
if self.k() > self.n() {
self.done = true;
} else {
self.first = false;
}
}

/// Increments indices representing the combination to advance to the next
/// (in lexicographic order by increasing sequence) combination. For example
/// if we have n=3 & k=2 then [0, 1] -> [0, 2] -> [0, 3] -> [1, 2] -> ...
fn increment_indices(&mut self) {
if self.indices.is_empty() {
self.done = true;
return;
}

// Scan from the end, looking for an index to increment
let mut i: usize = self.indices.len() - 1;

// Check if we need to consume more from the iterator
if self.indices[i] == self.pool.len() - 1 {
self.pool.get_next(); // may change pool size
}

while self.indices[i] == i + self.pool.len() - self.indices.len() {
if i > 0 {
i -= 1;
} else {
// Reached the last combination
self.done = true;
return;
}
}

// Increment index, and reset the ones to its right
self.indices[i] += 1;
for j in i + 1..self.indices.len() {
self.indices[j] = self.indices[j - 1] + 1;
}
}
}

impl<I> Iterator for Combinations<I>
Expand All @@ -104,40 +153,34 @@ where
type Item = Vec<I::Item>;
fn next(&mut self) -> Option<Self::Item> {
if self.first {
self.pool.prefill(self.k());
if self.k() > self.n() {
return None;
}
self.first = false;
} else if self.indices.is_empty() {
return None;
self.init()
} else {
// Scan from the end, looking for an index to increment
let mut i: usize = self.indices.len() - 1;
self.increment_indices()
}

// Check if we need to consume more from the iterator
if self.indices[i] == self.pool.len() - 1 {
self.pool.get_next(); // may change pool size
}
if self.done {
return None;
}

while self.indices[i] == i + self.pool.len() - self.indices.len() {
if i > 0 {
i -= 1;
} else {
// Reached the last combination
return None;
}
}
Some(self.indices.iter().map(|i| self.pool[*i].clone()).collect())
}

fn nth(&mut self, n: usize) -> Option<Self::Item> {
// Delegate initialisation work to next()
let first = self.next();

// Increment index, and reset the ones to its right
self.indices[i] += 1;
for j in i + 1..self.indices.len() {
self.indices[j] = self.indices[j - 1] + 1;
if n == 0 {
return first;
}

for _ in 0..(n - 1) {
self.increment_indices();
if self.done {
return None;
}
}

// Create result vector based on the indices
Some(self.indices.iter().map(|i| self.pool[*i].clone()).collect())
self.next()
}

fn size_hint(&self) -> (usize, Option<usize>) {
Expand Down

0 comments on commit 4cbb6d3

Please sign in to comment.