| 1 | use alloc::boxed::Box; |
| 2 | use alloc::vec::Vec; |
| 3 | use std::fmt; |
| 4 | use std::iter::once; |
| 5 | use std::iter::FusedIterator; |
| 6 | |
| 7 | use super::lazy_buffer::LazyBuffer; |
| 8 | use crate::size_hint::{self, SizeHint}; |
| 9 | |
| 10 | /// An iterator adaptor that iterates through all the `k`-permutations of the |
| 11 | /// elements from an iterator. |
| 12 | /// |
| 13 | /// See [`.permutations()`](crate::Itertools::permutations) for |
| 14 | /// more information. |
| 15 | #[must_use = "iterator adaptors are lazy and do nothing unless consumed" ] |
| 16 | pub struct Permutations<I: Iterator> { |
| 17 | vals: LazyBuffer<I>, |
| 18 | state: PermutationState, |
| 19 | } |
| 20 | |
| 21 | impl<I> Clone for Permutations<I> |
| 22 | where |
| 23 | I: Clone + Iterator, |
| 24 | I::Item: Clone, |
| 25 | { |
| 26 | clone_fields!(vals, state); |
| 27 | } |
| 28 | |
| 29 | #[derive (Clone, Debug)] |
| 30 | enum PermutationState { |
| 31 | /// No permutation generated yet. |
| 32 | Start { k: usize }, |
| 33 | /// Values from the iterator are not fully loaded yet so `n` is still unknown. |
| 34 | Buffered { k: usize, min_n: usize }, |
| 35 | /// All values from the iterator are known so `n` is known. |
| 36 | Loaded { |
| 37 | indices: Box<[usize]>, |
| 38 | cycles: Box<[usize]>, |
| 39 | }, |
| 40 | /// No permutation left to generate. |
| 41 | End, |
| 42 | } |
| 43 | |
| 44 | impl<I> fmt::Debug for Permutations<I> |
| 45 | where |
| 46 | I: Iterator + fmt::Debug, |
| 47 | I::Item: fmt::Debug, |
| 48 | { |
| 49 | debug_fmt_fields!(Permutations, vals, state); |
| 50 | } |
| 51 | |
| 52 | pub fn permutations<I: Iterator>(iter: I, k: usize) -> Permutations<I> { |
| 53 | Permutations { |
| 54 | vals: LazyBuffer::new(it:iter), |
| 55 | state: PermutationState::Start { k }, |
| 56 | } |
| 57 | } |
| 58 | |
| 59 | impl<I> Iterator for Permutations<I> |
| 60 | where |
| 61 | I: Iterator, |
| 62 | I::Item: Clone, |
| 63 | { |
| 64 | type Item = Vec<I::Item>; |
| 65 | |
| 66 | fn next(&mut self) -> Option<Self::Item> { |
| 67 | let Self { vals, state } = self; |
| 68 | match state { |
| 69 | PermutationState::Start { k: 0 } => { |
| 70 | *state = PermutationState::End; |
| 71 | Some(Vec::new()) |
| 72 | } |
| 73 | &mut PermutationState::Start { k } => { |
| 74 | vals.prefill(k); |
| 75 | if vals.len() != k { |
| 76 | *state = PermutationState::End; |
| 77 | return None; |
| 78 | } |
| 79 | *state = PermutationState::Buffered { k, min_n: k }; |
| 80 | Some(vals[0..k].to_vec()) |
| 81 | } |
| 82 | PermutationState::Buffered { ref k, min_n } => { |
| 83 | if vals.get_next() { |
| 84 | let item = (0..*k - 1) |
| 85 | .chain(once(*min_n)) |
| 86 | .map(|i| vals[i].clone()) |
| 87 | .collect(); |
| 88 | *min_n += 1; |
| 89 | Some(item) |
| 90 | } else { |
| 91 | let n = *min_n; |
| 92 | let prev_iteration_count = n - *k + 1; |
| 93 | let mut indices: Box<[_]> = (0..n).collect(); |
| 94 | let mut cycles: Box<[_]> = (n - k..n).rev().collect(); |
| 95 | // Advance the state to the correct point. |
| 96 | for _ in 0..prev_iteration_count { |
| 97 | if advance(&mut indices, &mut cycles) { |
| 98 | *state = PermutationState::End; |
| 99 | return None; |
| 100 | } |
| 101 | } |
| 102 | let item = indices[0..*k].iter().map(|&i| vals[i].clone()).collect(); |
| 103 | *state = PermutationState::Loaded { indices, cycles }; |
| 104 | Some(item) |
| 105 | } |
| 106 | } |
| 107 | PermutationState::Loaded { indices, cycles } => { |
| 108 | if advance(indices, cycles) { |
| 109 | *state = PermutationState::End; |
| 110 | return None; |
| 111 | } |
| 112 | let k = cycles.len(); |
| 113 | Some(indices[0..k].iter().map(|&i| vals[i].clone()).collect()) |
| 114 | } |
| 115 | PermutationState::End => None, |
| 116 | } |
| 117 | } |
| 118 | |
| 119 | fn count(self) -> usize { |
| 120 | let Self { vals, state } = self; |
| 121 | let n = vals.count(); |
| 122 | state.size_hint_for(n).1.unwrap() |
| 123 | } |
| 124 | |
| 125 | fn size_hint(&self) -> SizeHint { |
| 126 | let (mut low, mut upp) = self.vals.size_hint(); |
| 127 | low = self.state.size_hint_for(low).0; |
| 128 | upp = upp.and_then(|n| self.state.size_hint_for(n).1); |
| 129 | (low, upp) |
| 130 | } |
| 131 | } |
| 132 | |
| 133 | impl<I> FusedIterator for Permutations<I> |
| 134 | where |
| 135 | I: Iterator, |
| 136 | I::Item: Clone, |
| 137 | { |
| 138 | } |
| 139 | |
| 140 | fn advance(indices: &mut [usize], cycles: &mut [usize]) -> bool { |
| 141 | let n: usize = indices.len(); |
| 142 | let k: usize = cycles.len(); |
| 143 | // NOTE: if `cycles` are only zeros, then we reached the last permutation. |
| 144 | for i: usize in (0..k).rev() { |
| 145 | if cycles[i] == 0 { |
| 146 | cycles[i] = n - i - 1; |
| 147 | indices[i..].rotate_left(mid:1); |
| 148 | } else { |
| 149 | let swap_index: usize = n - cycles[i]; |
| 150 | indices.swap(a:i, b:swap_index); |
| 151 | cycles[i] -= 1; |
| 152 | return false; |
| 153 | } |
| 154 | } |
| 155 | true |
| 156 | } |
| 157 | |
| 158 | impl PermutationState { |
| 159 | fn size_hint_for(&self, n: usize) -> SizeHint { |
| 160 | // At the beginning, there are `n!/(n-k)!` items to come. |
| 161 | let at_start = |n, k| { |
| 162 | debug_assert!(n >= k); |
| 163 | let total = (n - k + 1..=n).try_fold(1usize, |acc, i| acc.checked_mul(i)); |
| 164 | (total.unwrap_or(usize::MAX), total) |
| 165 | }; |
| 166 | match *self { |
| 167 | Self::Start { k } if n < k => (0, Some(0)), |
| 168 | Self::Start { k } => at_start(n, k), |
| 169 | Self::Buffered { k, min_n } => { |
| 170 | // Same as `Start` minus the previously generated items. |
| 171 | size_hint::sub_scalar(at_start(n, k), min_n - k + 1) |
| 172 | } |
| 173 | Self::Loaded { |
| 174 | ref indices, |
| 175 | ref cycles, |
| 176 | } => { |
| 177 | let count = cycles.iter().enumerate().try_fold(0usize, |acc, (i, &c)| { |
| 178 | acc.checked_mul(indices.len() - i) |
| 179 | .and_then(|count| count.checked_add(c)) |
| 180 | }); |
| 181 | (count.unwrap_or(usize::MAX), count) |
| 182 | } |
| 183 | Self::End => (0, Some(0)), |
| 184 | } |
| 185 | } |
| 186 | } |
| 187 | |