diff options
| author | Scott McMurray <scottmcm@users.noreply.github.com> | 2018-03-01 01:57:25 -0800 |
|---|---|---|
| committer | Scott McMurray <scottmcm@users.noreply.github.com> | 2018-03-01 01:57:25 -0800 |
| commit | 70d5a4600b21451aa98b447cd59384d86e2eadce (patch) | |
| tree | 5d1a396c78855cb8a510c3e69282f129c19cb4bf | |
| parent | a85417f5938023d1491b44d94da705f539bb8b17 (diff) | |
| download | rust-70d5a4600b21451aa98b447cd59384d86e2eadce.tar.gz rust-70d5a4600b21451aa98b447cd59384d86e2eadce.zip | |
Specialize Zip::nth for TrustedRandomAccess
Makes the bench asked about on URLO 58x faster :)
| -rw-r--r-- | src/libcore/benches/iter.rs | 29 | ||||
| -rw-r--r-- | src/libcore/iter/mod.rs | 38 | ||||
| -rw-r--r-- | src/libcore/tests/iter.rs | 17 |
3 files changed, 84 insertions, 0 deletions
diff --git a/src/libcore/benches/iter.rs b/src/libcore/benches/iter.rs index b284d855c45..6c597301ac2 100644 --- a/src/libcore/benches/iter.rs +++ b/src/libcore/benches/iter.rs @@ -281,3 +281,32 @@ bench_sums! { bench_take_while_chain_ref_sum, (0i64..1000000).chain(1000000..).take_while(|&x| x < 1111111) } + +// Checks whether Skip<Zip<A,B>> is as fast as Zip<Skip<A>, Skip<B>>, from +// https://users.rust-lang.org/t/performance-difference-between-iterator-zip-and-skip-order/15743 +#[bench] +fn bench_zip_then_skip(b: &mut Bencher) { + let v: Vec<_> = (0..100_000).collect(); + let t: Vec<_> = (0..100_000).collect(); + + b.iter(|| { + let s = v.iter().zip(t.iter()).skip(10000) + .take_while(|t| *t.0 < 10100) + .map(|(a, b)| *a + *b) + .sum::<u64>(); + assert_eq!(s, 2009900); + }); +} +#[bench] +fn bench_skip_then_zip(b: &mut Bencher) { + let v: Vec<_> = (0..100_000).collect(); + let t: Vec<_> = (0..100_000).collect(); + + b.iter(|| { + let s = v.iter().skip(10000).zip(t.iter().skip(10000)) + .take_while(|t| *t.0 < 10100) + .map(|(a, b)| *a + *b) + .sum::<u64>(); + assert_eq!(s, 2009900); + }); +} diff --git a/src/libcore/iter/mod.rs b/src/libcore/iter/mod.rs index 623cad754dd..533ff388b76 100644 --- a/src/libcore/iter/mod.rs +++ b/src/libcore/iter/mod.rs @@ -1045,6 +1045,11 @@ impl<A, B> Iterator for Zip<A, B> where A: Iterator, B: Iterator fn size_hint(&self) -> (usize, Option<usize>) { ZipImpl::size_hint(self) } + + #[inline] + fn nth(&mut self, n: usize) -> Option<Self::Item> { + ZipImpl::nth(self, n) + } } #[stable(feature = "rust1", since = "1.0.0")] @@ -1065,6 +1070,14 @@ trait ZipImpl<A, B> { fn new(a: A, b: B) -> Self; fn next(&mut self) -> Option<Self::Item>; fn size_hint(&self) -> (usize, Option<usize>); + fn nth(&mut self, n: usize) -> Option<Self::Item>; + fn super_nth(&mut self, mut n: usize) -> Option<Self::Item> { + while let Some(x) = self.next() { + if n == 0 { return Some(x) } + n -= 1; + } + None + } fn next_back(&mut self) -> Option<Self::Item> where A: DoubleEndedIterator + ExactSizeIterator, B: DoubleEndedIterator + ExactSizeIterator; @@ -1095,6 +1108,12 @@ impl<A, B> ZipImpl<A, B> for Zip<A, B> } #[inline] + default fn nth(&mut self, n: usize) -> Option<Self::Item> + { + self.super_nth(n) + } + + #[inline] default fn next_back(&mut self) -> Option<(A::Item, B::Item)> where A: DoubleEndedIterator + ExactSizeIterator, B: DoubleEndedIterator + ExactSizeIterator @@ -1175,6 +1194,25 @@ impl<A, B> ZipImpl<A, B> for Zip<A, B> } #[inline] + fn nth(&mut self, n: usize) -> Option<Self::Item> + { + let delta = cmp::min(n, self.len - self.index); + let end = self.index + delta; + while self.index < end { + let i = self.index; + self.index += 1; + if A::may_have_side_effect() { + unsafe { self.a.get_unchecked(i); } + } + if B::may_have_side_effect() { + unsafe { self.b.get_unchecked(i); } + } + } + + self.super_nth(n - delta) + } + + #[inline] fn next_back(&mut self) -> Option<(A::Item, B::Item)> where A: DoubleEndedIterator + ExactSizeIterator, B: DoubleEndedIterator + ExactSizeIterator diff --git a/src/libcore/tests/iter.rs b/src/libcore/tests/iter.rs index edd75f7795e..5e2fef95d26 100644 --- a/src/libcore/tests/iter.rs +++ b/src/libcore/tests/iter.rs @@ -145,6 +145,23 @@ fn test_iterator_chain_find() { } #[test] +fn test_zip_nth() { + let xs = [0, 1, 2, 4, 5]; + let ys = [10, 11, 12]; + + let mut it = xs.iter().zip(&ys); + assert_eq!(it.nth(0), Some((&0, &10))); + assert_eq!(it.nth(1), Some((&2, &12))); + assert_eq!(it.nth(0), None); + + let mut it = xs.iter().zip(&ys); + assert_eq!(it.nth(3), None); + + let mut it = ys.iter().zip(&xs); + assert_eq!(it.nth(3), None); +} + +#[test] fn test_iterator_step_by() { // Identity let mut it = (0..).step_by(1).take(3); |
