core/iter/adapters/
take.rs

1use crate::cmp;
2#[cfg(not(feature = "ferrocene_certified"))]
3use crate::iter::adapters::SourceIter;
4#[cfg(not(feature = "ferrocene_certified"))]
5use crate::iter::{FusedIterator, InPlaceIterable, TrustedFused, TrustedLen, TrustedRandomAccess};
6use crate::num::NonZero;
7use crate::ops::{ControlFlow, Try};
8
9/// An iterator that only iterates over the first `n` iterations of `iter`.
10///
11/// This `struct` is created by the [`take`] method on [`Iterator`]. See its
12/// documentation for more.
13///
14/// [`take`]: Iterator::take
15/// [`Iterator`]: trait.Iterator.html
16#[cfg_attr(not(feature = "ferrocene_certified"), derive(Clone, Debug))]
17#[must_use = "iterators are lazy and do nothing unless consumed"]
18#[stable(feature = "rust1", since = "1.0.0")]
19pub struct Take<I> {
20    iter: I,
21    n: usize,
22}
23
24impl<I> Take<I> {
25    pub(in crate::iter) fn new(iter: I, n: usize) -> Take<I> {
26        Take { iter, n }
27    }
28}
29
30#[stable(feature = "rust1", since = "1.0.0")]
31impl<I> Iterator for Take<I>
32where
33    I: Iterator,
34{
35    type Item = <I as Iterator>::Item;
36
37    #[inline]
38    fn next(&mut self) -> Option<<I as Iterator>::Item> {
39        if self.n != 0 {
40            self.n -= 1;
41            self.iter.next()
42        } else {
43            None
44        }
45    }
46
47    #[inline]
48    fn nth(&mut self, n: usize) -> Option<I::Item> {
49        if self.n > n {
50            self.n -= n + 1;
51            self.iter.nth(n)
52        } else {
53            if self.n > 0 {
54                self.iter.nth(self.n - 1);
55                self.n = 0;
56            }
57            None
58        }
59    }
60
61    #[inline]
62    fn size_hint(&self) -> (usize, Option<usize>) {
63        if self.n == 0 {
64            return (0, Some(0));
65        }
66
67        let (lower, upper) = self.iter.size_hint();
68
69        let lower = cmp::min(lower, self.n);
70
71        let upper = match upper {
72            Some(x) if x < self.n => Some(x),
73            _ => Some(self.n),
74        };
75
76        (lower, upper)
77    }
78
79    #[inline]
80    fn try_fold<Acc, Fold, R>(&mut self, init: Acc, fold: Fold) -> R
81    where
82        Fold: FnMut(Acc, Self::Item) -> R,
83        R: Try<Output = Acc>,
84    {
85        fn check<'a, T, Acc, R: Try<Output = Acc>>(
86            n: &'a mut usize,
87            mut fold: impl FnMut(Acc, T) -> R + 'a,
88        ) -> impl FnMut(Acc, T) -> ControlFlow<R, Acc> + 'a {
89            move |acc, x| {
90                *n -= 1;
91                let r = fold(acc, x);
92                if *n == 0 { ControlFlow::Break(r) } else { ControlFlow::from_try(r) }
93            }
94        }
95
96        if self.n == 0 {
97            try { init }
98        } else {
99            let n = &mut self.n;
100            self.iter.try_fold(init, check(n, fold)).into_try()
101        }
102    }
103
104    #[inline]
105    fn fold<B, F>(self, init: B, f: F) -> B
106    where
107        Self: Sized,
108        F: FnMut(B, Self::Item) -> B,
109    {
110        Self::spec_fold(self, init, f)
111    }
112
113    #[inline]
114    fn for_each<F: FnMut(Self::Item)>(self, f: F) {
115        Self::spec_for_each(self, f)
116    }
117
118    #[inline]
119    #[rustc_inherit_overflow_checks]
120    fn advance_by(&mut self, n: usize) -> Result<(), NonZero<usize>> {
121        let min = self.n.min(n);
122        let rem = match self.iter.advance_by(min) {
123            Ok(()) => 0,
124            Err(rem) => rem.get(),
125        };
126        let advanced = min - rem;
127        self.n -= advanced;
128        NonZero::new(n - advanced).map_or(Ok(()), Err)
129    }
130}
131
132#[unstable(issue = "none", feature = "inplace_iteration")]
133#[cfg(not(feature = "ferrocene_certified"))]
134unsafe impl<I> SourceIter for Take<I>
135where
136    I: SourceIter,
137{
138    type Source = I::Source;
139
140    #[inline]
141    unsafe fn as_inner(&mut self) -> &mut I::Source {
142        // SAFETY: unsafe function forwarding to unsafe function with the same requirements
143        unsafe { SourceIter::as_inner(&mut self.iter) }
144    }
145}
146
147#[unstable(issue = "none", feature = "inplace_iteration")]
148#[cfg(not(feature = "ferrocene_certified"))]
149unsafe impl<I: InPlaceIterable> InPlaceIterable for Take<I> {
150    const EXPAND_BY: Option<NonZero<usize>> = I::EXPAND_BY;
151    const MERGE_BY: Option<NonZero<usize>> = I::MERGE_BY;
152}
153
154#[stable(feature = "double_ended_take_iterator", since = "1.38.0")]
155#[cfg(not(feature = "ferrocene_certified"))]
156impl<I> DoubleEndedIterator for Take<I>
157where
158    I: DoubleEndedIterator + ExactSizeIterator,
159{
160    #[inline]
161    fn next_back(&mut self) -> Option<Self::Item> {
162        if self.n == 0 {
163            None
164        } else {
165            let n = self.n;
166            self.n -= 1;
167            self.iter.nth_back(self.iter.len().saturating_sub(n))
168        }
169    }
170
171    #[inline]
172    fn nth_back(&mut self, n: usize) -> Option<Self::Item> {
173        let len = self.iter.len();
174        if self.n > n {
175            let m = len.saturating_sub(self.n) + n;
176            self.n -= n + 1;
177            self.iter.nth_back(m)
178        } else {
179            if len > 0 {
180                self.iter.nth_back(len - 1);
181            }
182            None
183        }
184    }
185
186    #[inline]
187    fn try_rfold<Acc, Fold, R>(&mut self, init: Acc, fold: Fold) -> R
188    where
189        Self: Sized,
190        Fold: FnMut(Acc, Self::Item) -> R,
191        R: Try<Output = Acc>,
192    {
193        if self.n == 0 {
194            try { init }
195        } else {
196            let len = self.iter.len();
197            if len > self.n && self.iter.nth_back(len - self.n - 1).is_none() {
198                try { init }
199            } else {
200                self.iter.try_rfold(init, fold)
201            }
202        }
203    }
204
205    #[inline]
206    fn rfold<Acc, Fold>(mut self, init: Acc, fold: Fold) -> Acc
207    where
208        Self: Sized,
209        Fold: FnMut(Acc, Self::Item) -> Acc,
210    {
211        if self.n == 0 {
212            init
213        } else {
214            let len = self.iter.len();
215            if len > self.n && self.iter.nth_back(len - self.n - 1).is_none() {
216                init
217            } else {
218                self.iter.rfold(init, fold)
219            }
220        }
221    }
222
223    #[inline]
224    #[rustc_inherit_overflow_checks]
225    fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero<usize>> {
226        // The amount by which the inner iterator needs to be shortened for it to be
227        // at most as long as the take() amount.
228        let trim_inner = self.iter.len().saturating_sub(self.n);
229        // The amount we need to advance inner to fulfill the caller's request.
230        // take(), advance_by() and len() all can be at most usize, so we don't have to worry
231        // about having to advance more than usize::MAX here.
232        let advance_by = trim_inner.saturating_add(n);
233
234        let remainder = match self.iter.advance_back_by(advance_by) {
235            Ok(()) => 0,
236            Err(rem) => rem.get(),
237        };
238        let advanced_by_inner = advance_by - remainder;
239        let advanced_by = advanced_by_inner - trim_inner;
240        self.n -= advanced_by;
241        NonZero::new(n - advanced_by).map_or(Ok(()), Err)
242    }
243}
244
245#[stable(feature = "rust1", since = "1.0.0")]
246#[cfg(not(feature = "ferrocene_certified"))]
247impl<I> ExactSizeIterator for Take<I> where I: ExactSizeIterator {}
248
249#[stable(feature = "fused", since = "1.26.0")]
250#[cfg(not(feature = "ferrocene_certified"))]
251impl<I> FusedIterator for Take<I> where I: FusedIterator {}
252
253#[unstable(issue = "none", feature = "trusted_fused")]
254#[cfg(not(feature = "ferrocene_certified"))]
255unsafe impl<I: TrustedFused> TrustedFused for Take<I> {}
256
257#[unstable(feature = "trusted_len", issue = "37572")]
258#[cfg(not(feature = "ferrocene_certified"))]
259unsafe impl<I: TrustedLen> TrustedLen for Take<I> {}
260
261trait SpecTake: Iterator {
262    fn spec_fold<B, F>(self, init: B, f: F) -> B
263    where
264        Self: Sized,
265        F: FnMut(B, Self::Item) -> B;
266
267    fn spec_for_each<F: FnMut(Self::Item)>(self, f: F);
268}
269
270impl<I: Iterator> SpecTake for Take<I> {
271    #[inline]
272    default fn spec_fold<B, F>(mut self, init: B, f: F) -> B
273    where
274        Self: Sized,
275        F: FnMut(B, Self::Item) -> B,
276    {
277        use crate::ops::NeverShortCircuit;
278        self.try_fold(init, NeverShortCircuit::wrap_mut_2(f)).0
279    }
280
281    #[inline]
282    default fn spec_for_each<F: FnMut(Self::Item)>(mut self, f: F) {
283        // The default implementation would use a unit accumulator, so we can
284        // avoid a stateful closure by folding over the remaining number
285        // of items we wish to return instead.
286        fn check<'a, Item>(
287            mut action: impl FnMut(Item) + 'a,
288        ) -> impl FnMut(usize, Item) -> Option<usize> + 'a {
289            move |more, x| {
290                action(x);
291                more.checked_sub(1)
292            }
293        }
294
295        let remaining = self.n;
296        if remaining > 0 {
297            self.iter.try_fold(remaining - 1, check(f));
298        }
299    }
300}
301
302#[cfg(not(feature = "ferrocene_certified"))]
303impl<I: Iterator + TrustedRandomAccess> SpecTake for Take<I> {
304    #[inline]
305    fn spec_fold<B, F>(mut self, init: B, mut f: F) -> B
306    where
307        Self: Sized,
308        F: FnMut(B, Self::Item) -> B,
309    {
310        let mut acc = init;
311        let end = self.n.min(self.iter.size());
312        for i in 0..end {
313            // SAFETY: i < end <= self.iter.size() and we discard the iterator at the end
314            let val = unsafe { self.iter.__iterator_get_unchecked(i) };
315            acc = f(acc, val);
316        }
317        acc
318    }
319
320    #[inline]
321    fn spec_for_each<F: FnMut(Self::Item)>(mut self, mut f: F) {
322        let end = self.n.min(self.iter.size());
323        for i in 0..end {
324            // SAFETY: i < end <= self.iter.size() and we discard the iterator at the end
325            let val = unsafe { self.iter.__iterator_get_unchecked(i) };
326            f(val);
327        }
328    }
329}
330
331#[stable(feature = "exact_size_take_repeat", since = "1.82.0")]
332#[cfg(not(feature = "ferrocene_certified"))]
333impl<T: Clone> DoubleEndedIterator for Take<crate::iter::Repeat<T>> {
334    #[inline]
335    fn next_back(&mut self) -> Option<Self::Item> {
336        self.next()
337    }
338
339    #[inline]
340    fn nth_back(&mut self, n: usize) -> Option<Self::Item> {
341        self.nth(n)
342    }
343
344    #[inline]
345    fn try_rfold<Acc, Fold, R>(&mut self, init: Acc, fold: Fold) -> R
346    where
347        Self: Sized,
348        Fold: FnMut(Acc, Self::Item) -> R,
349        R: Try<Output = Acc>,
350    {
351        self.try_fold(init, fold)
352    }
353
354    #[inline]
355    fn rfold<Acc, Fold>(self, init: Acc, fold: Fold) -> Acc
356    where
357        Self: Sized,
358        Fold: FnMut(Acc, Self::Item) -> Acc,
359    {
360        self.fold(init, fold)
361    }
362
363    #[inline]
364    #[rustc_inherit_overflow_checks]
365    fn advance_back_by(&mut self, n: usize) -> Result<(), NonZero<usize>> {
366        self.advance_by(n)
367    }
368}
369
370// Note: It may be tempting to impl DoubleEndedIterator for Take<RepeatWith>.
371// One must fight that temptation since such implementation wouldn’t be correct
372// because we have no way to return value of nth invocation of repeater followed
373// by n-1st without remembering all results.
374
375#[stable(feature = "exact_size_take_repeat", since = "1.82.0")]
376#[cfg(not(feature = "ferrocene_certified"))]
377impl<T: Clone> ExactSizeIterator for Take<crate::iter::Repeat<T>> {
378    fn len(&self) -> usize {
379        self.n
380    }
381}
382
383#[stable(feature = "exact_size_take_repeat", since = "1.82.0")]
384#[cfg(not(feature = "ferrocene_certified"))]
385impl<F: FnMut() -> A, A> ExactSizeIterator for Take<crate::iter::RepeatWith<F>> {
386    fn len(&self) -> usize {
387        self.n
388    }
389}