Skip to content

Commit 8bd4ed9

Browse files
authored
Rollup merge of #76909 - timvermeulen:advance_by, r=Amanieu
Add Iterator::advance_by and DoubleEndedIterator::advance_back_by This PR adds the iterator method ```rust fn advance_by(&mut self, n: usize) -> Result<(), usize> ``` that advances the iterator by `n` elements, returning `Ok(())` if this succeeds or `Err(len)` if the length of the iterator was less than `n`. Currently `Iterator::nth` is the method to override for efficiently advancing an iterator by multiple elements at once. `advance_by` is superior for this purpose because - it's simpler to implement: instead of advancing the iterator and producing the next element you only need to advance the iterator - it composes better: iterators like `Chain` and `FlatMap` can implement `advance_by` in terms of `advance_by` on their inner iterators, but they cannot implement `nth` in terms of `nth` on their inner iterators (see #60395) - the default implementation of `nth` can trivially be implemented in terms of `advance_by` and `next`, which this PR also does This PR also adds `DoubleEndedIterator::advance_back_by` for all the same reasons. I'll make a tracking issue if it's decided this is worth merging. Also let me know if anything can be improved, this went through several iterations so there might very well still be room for improvement (especially in the doc comments). I've written overrides of these methods for most iterators that already override `nth`/`nth_back`, but those still need tests so I'll add them in a later PR. cc @cuviper @scottmcm @Amanieu
2 parents ef663a8 + ecacc75 commit 8bd4ed9

File tree

5 files changed

+161
-16
lines changed

5 files changed

+161
-16
lines changed

library/core/src/iter/adapters/mod.rs

+10
Original file line numberDiff line numberDiff line change
@@ -124,6 +124,11 @@ where
124124
self.iter.size_hint()
125125
}
126126

127+
#[inline]
128+
fn advance_by(&mut self, n: usize) -> Result<(), usize> {
129+
self.iter.advance_back_by(n)
130+
}
131+
127132
#[inline]
128133
fn nth(&mut self, n: usize) -> Option<<I as Iterator>::Item> {
129134
self.iter.nth_back(n)
@@ -164,6 +169,11 @@ where
164169
self.iter.next()
165170
}
166171

172+
#[inline]
173+
fn advance_back_by(&mut self, n: usize) -> Result<(), usize> {
174+
self.iter.advance_by(n)
175+
}
176+
167177
#[inline]
168178
fn nth_back(&mut self, n: usize) -> Option<<I as Iterator>::Item> {
169179
self.iter.nth(n)

library/core/src/iter/traits/double_ended.rs

+46-8
Original file line numberDiff line numberDiff line change
@@ -91,6 +91,46 @@ pub trait DoubleEndedIterator: Iterator {
9191
#[stable(feature = "rust1", since = "1.0.0")]
9292
fn next_back(&mut self) -> Option<Self::Item>;
9393

94+
/// Advances the iterator from the back by `n` elements.
95+
///
96+
/// `advance_back_by` is the reverse version of [`advance_by`]. This method will
97+
/// eagerly skip `n` elements starting from the back by calling [`next_back`] up
98+
/// to `n` times until [`None`] is encountered.
99+
///
100+
/// `advance_back_by(n)` will return [`Ok(())`] if the iterator successfully advances by
101+
/// `n` elements, or [`Err(k)`] if [`None`] is encountered, where `k` is the number of
102+
/// elements the iterator is advanced by before running out of elements (i.e. the length
103+
/// of the iterator). Note that `k` is always less than `n`.
104+
///
105+
/// Calling `advance_back_by(0)` does not consume any elements and always returns [`Ok(())`].
106+
///
107+
/// [`advance_by`]: Iterator::advance_by
108+
/// [`next_back`]: DoubleEndedIterator::next_back
109+
///
110+
/// # Examples
111+
///
112+
/// Basic usage:
113+
///
114+
/// ```
115+
/// #![feature(iter_advance_by)]
116+
///
117+
/// let a = [3, 4, 5, 6];
118+
/// let mut iter = a.iter();
119+
///
120+
/// assert_eq!(iter.advance_back_by(2), Ok(()));
121+
/// assert_eq!(iter.next_back(), Some(&4));
122+
/// assert_eq!(iter.advance_back_by(0), Ok(()));
123+
/// assert_eq!(iter.advance_back_by(100), Err(1)); // only `&3` was skipped
124+
/// ```
125+
#[inline]
126+
#[unstable(feature = "iter_advance_by", reason = "recently added", issue = "none")]
127+
fn advance_back_by(&mut self, n: usize) -> Result<(), usize> {
128+
for i in 0..n {
129+
self.next_back().ok_or(i)?;
130+
}
131+
Ok(())
132+
}
133+
94134
/// Returns the `n`th element from the end of the iterator.
95135
///
96136
/// This is essentially the reversed version of [`Iterator::nth()`].
@@ -134,14 +174,9 @@ pub trait DoubleEndedIterator: Iterator {
134174
/// ```
135175
#[inline]
136176
#[stable(feature = "iter_nth_back", since = "1.37.0")]
137-
fn nth_back(&mut self, mut n: usize) -> Option<Self::Item> {
138-
for x in self.rev() {
139-
if n == 0 {
140-
return Some(x);
141-
}
142-
n -= 1;
143-
}
144-
None
177+
fn nth_back(&mut self, n: usize) -> Option<Self::Item> {
178+
self.advance_back_by(n).ok()?;
179+
self.next_back()
145180
}
146181

147182
/// This is the reverse version of [`Iterator::try_fold()`]: it takes
@@ -318,6 +353,9 @@ impl<'a, I: DoubleEndedIterator + ?Sized> DoubleEndedIterator for &'a mut I {
318353
fn next_back(&mut self) -> Option<I::Item> {
319354
(**self).next_back()
320355
}
356+
fn advance_back_by(&mut self, n: usize) -> Result<(), usize> {
357+
(**self).advance_back_by(n)
358+
}
321359
fn nth_back(&mut self, n: usize) -> Option<I::Item> {
322360
(**self).nth_back(n)
323361
}

library/core/src/iter/traits/iterator.rs

+44-8
Original file line numberDiff line numberDiff line change
@@ -284,6 +284,44 @@ pub trait Iterator {
284284
self.fold(None, some)
285285
}
286286

287+
/// Advances the iterator by `n` elements.
288+
///
289+
/// This method will eagerly skip `n` elements by calling [`next`] up to `n`
290+
/// times until [`None`] is encountered.
291+
///
292+
/// `advance_by(n)` will return [`Ok(())`] if the iterator successfully advances by
293+
/// `n` elements, or [`Err(k)`] if [`None`] is encountered, where `k` is the number
294+
/// of elements the iterator is advanced by before running out of elements (i.e. the
295+
/// length of the iterator). Note that `k` is always less than `n`.
296+
///
297+
/// Calling `advance_by(0)` does not consume any elements and always returns [`Ok(())`].
298+
///
299+
/// [`next`]: Iterator::next
300+
///
301+
/// # Examples
302+
///
303+
/// Basic usage:
304+
///
305+
/// ```
306+
/// #![feature(iter_advance_by)]
307+
///
308+
/// let a = [1, 2, 3, 4];
309+
/// let mut iter = a.iter();
310+
///
311+
/// assert_eq!(iter.advance_by(2), Ok(()));
312+
/// assert_eq!(iter.next(), Some(&3));
313+
/// assert_eq!(iter.advance_by(0), Ok(()));
314+
/// assert_eq!(iter.advance_by(100), Err(1)); // only `&4` was skipped
315+
/// ```
316+
#[inline]
317+
#[unstable(feature = "iter_advance_by", reason = "recently added", issue = "none")]
318+
fn advance_by(&mut self, n: usize) -> Result<(), usize> {
319+
for i in 0..n {
320+
self.next().ok_or(i)?;
321+
}
322+
Ok(())
323+
}
324+
287325
/// Returns the `n`th element of the iterator.
288326
///
289327
/// Like most indexing operations, the count starts from zero, so `nth(0)`
@@ -325,14 +363,9 @@ pub trait Iterator {
325363
/// ```
326364
#[inline]
327365
#[stable(feature = "rust1", since = "1.0.0")]
328-
fn nth(&mut self, mut n: usize) -> Option<Self::Item> {
329-
while let Some(x) = self.next() {
330-
if n == 0 {
331-
return Some(x);
332-
}
333-
n -= 1;
334-
}
335-
None
366+
fn nth(&mut self, n: usize) -> Option<Self::Item> {
367+
self.advance_by(n).ok()?;
368+
self.next()
336369
}
337370

338371
/// Creates an iterator starting at the same point, but stepping by
@@ -3265,6 +3298,9 @@ impl<I: Iterator + ?Sized> Iterator for &mut I {
32653298
fn size_hint(&self) -> (usize, Option<usize>) {
32663299
(**self).size_hint()
32673300
}
3301+
fn advance_by(&mut self, n: usize) -> Result<(), usize> {
3302+
(**self).advance_by(n)
3303+
}
32683304
fn nth(&mut self, n: usize) -> Option<Self::Item> {
32693305
(**self).nth(n)
32703306
}

library/core/tests/iter.rs

+60
Original file line numberDiff line numberDiff line change
@@ -1570,6 +1570,66 @@ fn test_iterator_rev_nth() {
15701570
assert_eq!(v.iter().rev().nth(v.len()), None);
15711571
}
15721572

1573+
#[test]
1574+
fn test_iterator_advance_by() {
1575+
let v: &[_] = &[0, 1, 2, 3, 4];
1576+
1577+
for i in 0..v.len() {
1578+
let mut iter = v.iter();
1579+
assert_eq!(iter.advance_by(i), Ok(()));
1580+
assert_eq!(iter.next().unwrap(), &v[i]);
1581+
assert_eq!(iter.advance_by(100), Err(v.len() - 1 - i));
1582+
}
1583+
1584+
assert_eq!(v.iter().advance_by(v.len()), Ok(()));
1585+
assert_eq!(v.iter().advance_by(100), Err(v.len()));
1586+
}
1587+
1588+
#[test]
1589+
fn test_iterator_advance_back_by() {
1590+
let v: &[_] = &[0, 1, 2, 3, 4];
1591+
1592+
for i in 0..v.len() {
1593+
let mut iter = v.iter();
1594+
assert_eq!(iter.advance_back_by(i), Ok(()));
1595+
assert_eq!(iter.next_back().unwrap(), &v[v.len() - 1 - i]);
1596+
assert_eq!(iter.advance_back_by(100), Err(v.len() - 1 - i));
1597+
}
1598+
1599+
assert_eq!(v.iter().advance_back_by(v.len()), Ok(()));
1600+
assert_eq!(v.iter().advance_back_by(100), Err(v.len()));
1601+
}
1602+
1603+
#[test]
1604+
fn test_iterator_rev_advance_by() {
1605+
let v: &[_] = &[0, 1, 2, 3, 4];
1606+
1607+
for i in 0..v.len() {
1608+
let mut iter = v.iter().rev();
1609+
assert_eq!(iter.advance_by(i), Ok(()));
1610+
assert_eq!(iter.next().unwrap(), &v[v.len() - 1 - i]);
1611+
assert_eq!(iter.advance_by(100), Err(v.len() - 1 - i));
1612+
}
1613+
1614+
assert_eq!(v.iter().rev().advance_by(v.len()), Ok(()));
1615+
assert_eq!(v.iter().rev().advance_by(100), Err(v.len()));
1616+
}
1617+
1618+
#[test]
1619+
fn test_iterator_rev_advance_back_by() {
1620+
let v: &[_] = &[0, 1, 2, 3, 4];
1621+
1622+
for i in 0..v.len() {
1623+
let mut iter = v.iter().rev();
1624+
assert_eq!(iter.advance_back_by(i), Ok(()));
1625+
assert_eq!(iter.next_back().unwrap(), &v[i]);
1626+
assert_eq!(iter.advance_back_by(100), Err(v.len() - 1 - i));
1627+
}
1628+
1629+
assert_eq!(v.iter().rev().advance_back_by(v.len()), Ok(()));
1630+
assert_eq!(v.iter().rev().advance_back_by(100), Err(v.len()));
1631+
}
1632+
15731633
#[test]
15741634
fn test_iterator_last() {
15751635
let v: &[_] = &[0, 1, 2, 3, 4];

library/core/tests/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -41,6 +41,7 @@
4141
#![feature(slice_partition_dedup)]
4242
#![feature(int_error_matching)]
4343
#![feature(array_value_iter)]
44+
#![feature(iter_advance_by)]
4445
#![feature(iter_partition_in_place)]
4546
#![feature(iter_is_partitioned)]
4647
#![feature(iter_order_by)]

0 commit comments

Comments
 (0)