core/iter/adapters/
cycle.rs1use crate::iter::FusedIterator;
2use crate::num::NonZero;
3use crate::ops::Try;
4
5#[derive(Clone, Debug)]
13#[must_use = "iterators are lazy and do nothing unless consumed"]
14#[stable(feature = "rust1", since = "1.0.0")]
15pub struct Cycle<I> {
16 orig: I,
17 iter: I,
18}
19
20#[rustc_const_unstable(feature = "const_iter", issue = "92476")]
21const impl<I: [const] Clone> Cycle<I> {
22 pub(in crate::iter) fn new(iter: I) -> Cycle<I> {
23 Cycle { orig: iter.clone(), iter }
24 }
25}
26
27#[stable(feature = "rust1", since = "1.0.0")]
28impl<I> Iterator for Cycle<I>
29where
30 I: Clone + Iterator,
31{
32 type Item = <I as Iterator>::Item;
33
34 #[inline]
35 fn next(&mut self) -> Option<<I as Iterator>::Item> {
36 match self.iter.next() {
37 None => {
38 self.iter = self.orig.clone();
39 self.iter.next()
40 }
41 y => y,
42 }
43 }
44
45 #[inline]
46 fn size_hint(&self) -> (usize, Option<usize>) {
47 match self.orig.size_hint() {
49 sz @ (0, Some(0)) => sz,
50 (0, _) => (0, None),
51 _ => (usize::MAX, None),
52 }
53 }
54
55 #[inline]
56 fn try_fold<Acc, F, R>(&mut self, mut acc: Acc, mut f: F) -> R
57 where
58 F: FnMut(Acc, Self::Item) -> R,
59 R: Try<Output = Acc>,
60 {
61 acc = self.iter.try_fold(acc, &mut f)?;
64 self.iter = self.orig.clone();
65
66 let mut is_empty = true;
70 acc = self.iter.try_fold(acc, |acc, x| {
71 is_empty = false;
72 f(acc, x)
73 })?;
74
75 if is_empty {
76 return try { acc };
77 }
78
79 loop {
80 self.iter = self.orig.clone();
81 acc = self.iter.try_fold(acc, &mut f)?;
82 }
83 }
84
85 #[inline]
86 #[rustc_inherit_overflow_checks]
87 fn advance_by(&mut self, n: usize) -> Result<(), NonZero<usize>> {
88 let mut n = match self.iter.advance_by(n) {
89 Ok(()) => return Ok(()),
90 Err(rem) => rem.get(),
91 };
92
93 while n > 0 {
94 self.iter = self.orig.clone();
95 n = match self.iter.advance_by(n) {
96 Ok(()) => return Ok(()),
97 e @ Err(rem) if rem.get() == n => return e,
98 Err(rem) => rem.get(),
99 };
100 }
101
102 NonZero::new(n).map_or(Ok(()), Err)
103 }
104
105 }
108
109#[stable(feature = "fused", since = "1.26.0")]
110impl<I> FusedIterator for Cycle<I> where I: Clone + Iterator {}