1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
use std::collections::VecDeque;
use std::iter::Fuse;
pub fn lookahead<I>(iterable: I) -> Lookahead<I::IntoIter>
where
I: IntoIterator,
{
Lookahead {
iter: iterable.into_iter().fuse(),
queue: VecDeque::new(),
}
}
#[derive(Clone, Debug)]
pub struct Lookahead<I: Iterator> {
iter: Fuse<I>,
queue: VecDeque<I::Item>,
}
impl<I: Iterator> Lookahead<I> {
pub fn lookahead(&mut self, n: usize) -> Option<&I::Item> {
let enqueued = self.queue.len();
if n >= enqueued {
let iter = &mut self.iter;
let items = iter.take(n - enqueued + 1);
self.queue.extend(items);
}
self.queue.get(n)
}
}
impl<I> Iterator for Lookahead<I>
where
I: Iterator,
{
type Item = I::Item;
fn next(&mut self) -> Option<Self::Item> {
self.queue.pop_front().or_else(|| self.iter.next())
}
fn size_hint(&self) -> (usize, Option<usize>) {
let queued = self.queue.len();
let (lower, upper) = self.iter.size_hint();
(lower + queued, upper.map(|n| n + queued))
}
}
impl<I> ExactSizeIterator for Lookahead<I> where I: ExactSizeIterator {}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn zero() {
let inner = [1, 2].into_iter();
let mut iter = lookahead(inner);
assert_eq!(iter.lookahead(0), Some(&&1));
}
#[test]
fn one() {
let inner = [1, 2].into_iter();
let mut iter = lookahead(inner);
assert_eq!(iter.lookahead(1), Some(&&2));
}
#[test]
fn two() {
let inner = [1, 2].into_iter();
let mut iter = lookahead(inner);
assert_eq!(iter.lookahead(2), None);
}
#[test]
fn next() {
let inner = [1, 2].into_iter();
let mut iter = lookahead(inner);
let _ = iter.next();
assert_eq!(iter.lookahead(0), Some(&&2));
}
#[test]
fn size_hint() {
let inner = [1, 2].into_iter();
let mut iter = lookahead(inner);
assert_eq!(iter.size_hint(), (2, Some(2)));
let _ = iter.lookahead(1);
assert_eq!(iter.size_hint(), (2, Some(2)));
let _ = iter.next();
assert_eq!(iter.size_hint(), (1, Some(1)));
}
}