Skip to content

Commit b6ac411

Browse files
committed
Auto merge of #78015 - ssomers:btree_merge_mergers, r=Mark-Simulacrum
btree: merge the implementations of MergeIter Also remove the gratuitous Copy bounds. Same benchmark performance. r? `@Mark-Simulacrum`
2 parents 4760b8f + 2c5f64f commit b6ac411

File tree

4 files changed

+111
-97
lines changed

4 files changed

+111
-97
lines changed

library/alloc/src/collections/btree/map.rs

+11-24
Original file line numberDiff line numberDiff line change
@@ -2,13 +2,14 @@ use core::borrow::Borrow;
22
use core::cmp::Ordering;
33
use core::fmt::{self, Debug};
44
use core::hash::{Hash, Hasher};
5-
use core::iter::{FromIterator, FusedIterator, Peekable};
5+
use core::iter::{FromIterator, FusedIterator};
66
use core::marker::PhantomData;
77
use core::mem::{self, ManuallyDrop};
88
use core::ops::{Index, RangeBounds};
99
use core::ptr;
1010

1111
use super::borrow::DormantMutRef;
12+
use super::merge_iter::MergeIterInner;
1213
use super::node::{self, marker, ForceResult::*, Handle, NodeRef};
1314
use super::search::{self, SearchResult::*};
1415
use super::unwrap_unchecked;
@@ -458,10 +459,7 @@ impl<K: fmt::Debug, V: fmt::Debug> fmt::Debug for RangeMut<'_, K, V> {
458459
}
459460

460461
// An iterator for merging two sorted sequences into one
461-
struct MergeIter<K, V, I: Iterator<Item = (K, V)>> {
462-
left: Peekable<I>,
463-
right: Peekable<I>,
464-
}
462+
struct MergeIter<K, V, I: Iterator<Item = (K, V)>>(MergeIterInner<I>);
465463

466464
impl<K: Ord, V> BTreeMap<K, V> {
467465
/// Makes a new empty BTreeMap.
@@ -913,7 +911,7 @@ impl<K: Ord, V> BTreeMap<K, V> {
913911
// First, we merge `self` and `other` into a sorted sequence in linear time.
914912
let self_iter = mem::take(self).into_iter();
915913
let other_iter = mem::take(other).into_iter();
916-
let iter = MergeIter { left: self_iter.peekable(), right: other_iter.peekable() };
914+
let iter = MergeIter(MergeIterInner::new(self_iter, other_iter));
917915

918916
// Second, we build a tree from the sorted sequence in linear time.
919917
self.from_sorted_iter(iter);
@@ -2220,27 +2218,16 @@ impl<K, V> BTreeMap<K, V> {
22202218
}
22212219
}
22222220

2223-
impl<K: Ord, V, I: Iterator<Item = (K, V)>> Iterator for MergeIter<K, V, I> {
2221+
impl<K: Ord, V, I> Iterator for MergeIter<K, V, I>
2222+
where
2223+
I: Iterator<Item = (K, V)> + ExactSizeIterator + FusedIterator,
2224+
{
22242225
type Item = (K, V);
22252226

2227+
/// If two keys are equal, returns the key/value-pair from the right source.
22262228
fn next(&mut self) -> Option<(K, V)> {
2227-
let res = match (self.left.peek(), self.right.peek()) {
2228-
(Some(&(ref left_key, _)), Some(&(ref right_key, _))) => left_key.cmp(right_key),
2229-
(Some(_), None) => Ordering::Less,
2230-
(None, Some(_)) => Ordering::Greater,
2231-
(None, None) => return None,
2232-
};
2233-
2234-
// Check which elements comes first and only advance the corresponding iterator.
2235-
// If two keys are equal, take the value from `right`.
2236-
match res {
2237-
Ordering::Less => self.left.next(),
2238-
Ordering::Greater => self.right.next(),
2239-
Ordering::Equal => {
2240-
self.left.next();
2241-
self.right.next()
2242-
}
2243-
}
2229+
let (a_next, b_next) = self.0.nexts(|a: &(K, V), b: &(K, V)| K::cmp(&a.0, &b.0));
2230+
b_next.or(a_next)
22442231
}
22452232
}
22462233

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,96 @@
1+
use core::cmp::Ordering;
2+
use core::fmt::{self, Debug};
3+
use core::iter::FusedIterator;
4+
5+
/// Core of an iterator that merges the output of two ascending iterators,
6+
/// for instance a union or a symmetric difference.
7+
pub struct MergeIterInner<I>
8+
where
9+
I: Iterator,
10+
{
11+
a: I,
12+
b: I,
13+
peeked: Option<Peeked<I>>,
14+
}
15+
16+
/// Benchmarks faster than wrapping both iterators in a Peekable.
17+
#[derive(Clone, Debug)]
18+
enum Peeked<I: Iterator> {
19+
A(I::Item),
20+
B(I::Item),
21+
}
22+
23+
impl<I> Clone for MergeIterInner<I>
24+
where
25+
I: Clone + Iterator,
26+
I::Item: Clone,
27+
{
28+
fn clone(&self) -> Self {
29+
Self { a: self.a.clone(), b: self.b.clone(), peeked: self.peeked.clone() }
30+
}
31+
}
32+
33+
impl<I> Debug for MergeIterInner<I>
34+
where
35+
I: Iterator + Debug,
36+
I::Item: Debug,
37+
{
38+
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
39+
f.debug_tuple("MergeIterInner").field(&self.a).field(&self.b).finish()
40+
}
41+
}
42+
43+
impl<I> MergeIterInner<I>
44+
where
45+
I: ExactSizeIterator + FusedIterator,
46+
{
47+
/// Creates a new core for an iterator merging a pair of sources.
48+
pub fn new(a: I, b: I) -> Self {
49+
MergeIterInner { a, b, peeked: None }
50+
}
51+
52+
/// Returns the next pair of items stemming from the pair of sources
53+
/// being merged. If both returned options contain a value, that value
54+
/// is equal and occurs in both sources. If one of the returned options
55+
/// contains a value, that value doesn't occur in the other source.
56+
/// If neither returned option contains a value, iteration has finished
57+
/// and subsequent calls will return the same empty pair.
58+
pub fn nexts<Cmp: Fn(&I::Item, &I::Item) -> Ordering>(
59+
&mut self,
60+
cmp: Cmp,
61+
) -> (Option<I::Item>, Option<I::Item>) {
62+
let mut a_next;
63+
let mut b_next;
64+
match self.peeked.take() {
65+
Some(Peeked::A(next)) => {
66+
a_next = Some(next);
67+
b_next = self.b.next();
68+
}
69+
Some(Peeked::B(next)) => {
70+
b_next = Some(next);
71+
a_next = self.a.next();
72+
}
73+
None => {
74+
a_next = self.a.next();
75+
b_next = self.b.next();
76+
}
77+
}
78+
if let (Some(ref a1), Some(ref b1)) = (&a_next, &b_next) {
79+
match cmp(a1, b1) {
80+
Ordering::Less => self.peeked = b_next.take().map(Peeked::B),
81+
Ordering::Greater => self.peeked = a_next.take().map(Peeked::A),
82+
Ordering::Equal => (),
83+
}
84+
}
85+
(a_next, b_next)
86+
}
87+
88+
/// Returns a pair of upper bounds for the `size_hint` of the final iterator.
89+
pub fn lens(&self) -> (usize, usize) {
90+
match self.peeked {
91+
Some(Peeked::A(_)) => (1 + self.a.len(), self.b.len()),
92+
Some(Peeked::B(_)) => (self.a.len(), 1 + self.b.len()),
93+
_ => (self.a.len(), self.b.len()),
94+
}
95+
}
96+
}

library/alloc/src/collections/btree/mod.rs

+1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
mod borrow;
22
pub mod map;
3+
mod merge_iter;
34
mod navigate;
45
mod node;
56
mod remove;

library/alloc/src/collections/btree/set.rs

+3-73
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ use core::iter::{FromIterator, FusedIterator, Peekable};
99
use core::ops::{BitAnd, BitOr, BitXor, RangeBounds, Sub};
1010

1111
use super::map::{BTreeMap, Keys};
12+
use super::merge_iter::MergeIterInner;
1213
use super::Recover;
1314

1415
// FIXME(conventions): implement bounded iterators
@@ -114,77 +115,6 @@ pub struct Range<'a, T: 'a> {
114115
iter: super::map::Range<'a, T, ()>,
115116
}
116117

117-
/// Core of SymmetricDifference and Union.
118-
/// More efficient than btree.map.MergeIter,
119-
/// and crucially for SymmetricDifference, nexts() reports on both sides.
120-
#[derive(Clone)]
121-
struct MergeIterInner<I>
122-
where
123-
I: Iterator,
124-
I::Item: Copy,
125-
{
126-
a: I,
127-
b: I,
128-
peeked: Option<MergeIterPeeked<I>>,
129-
}
130-
131-
#[derive(Copy, Clone, Debug)]
132-
enum MergeIterPeeked<I: Iterator> {
133-
A(I::Item),
134-
B(I::Item),
135-
}
136-
137-
impl<I> MergeIterInner<I>
138-
where
139-
I: ExactSizeIterator + FusedIterator,
140-
I::Item: Copy + Ord,
141-
{
142-
fn new(a: I, b: I) -> Self {
143-
MergeIterInner { a, b, peeked: None }
144-
}
145-
146-
fn nexts(&mut self) -> (Option<I::Item>, Option<I::Item>) {
147-
let mut a_next = match self.peeked {
148-
Some(MergeIterPeeked::A(next)) => Some(next),
149-
_ => self.a.next(),
150-
};
151-
let mut b_next = match self.peeked {
152-
Some(MergeIterPeeked::B(next)) => Some(next),
153-
_ => self.b.next(),
154-
};
155-
let ord = match (a_next, b_next) {
156-
(None, None) => Equal,
157-
(_, None) => Less,
158-
(None, _) => Greater,
159-
(Some(a1), Some(b1)) => a1.cmp(&b1),
160-
};
161-
self.peeked = match ord {
162-
Less => b_next.take().map(MergeIterPeeked::B),
163-
Equal => None,
164-
Greater => a_next.take().map(MergeIterPeeked::A),
165-
};
166-
(a_next, b_next)
167-
}
168-
169-
fn lens(&self) -> (usize, usize) {
170-
match self.peeked {
171-
Some(MergeIterPeeked::A(_)) => (1 + self.a.len(), self.b.len()),
172-
Some(MergeIterPeeked::B(_)) => (self.a.len(), 1 + self.b.len()),
173-
_ => (self.a.len(), self.b.len()),
174-
}
175-
}
176-
}
177-
178-
impl<I> Debug for MergeIterInner<I>
179-
where
180-
I: Iterator + Debug,
181-
I::Item: Copy + Debug,
182-
{
183-
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
184-
f.debug_tuple("MergeIterInner").field(&self.a).field(&self.b).finish()
185-
}
186-
}
187-
188118
/// A lazy iterator producing elements in the difference of `BTreeSet`s.
189119
///
190120
/// This `struct` is created by the [`difference`] method on [`BTreeSet`].
@@ -1461,7 +1391,7 @@ impl<'a, T: Ord> Iterator for SymmetricDifference<'a, T> {
14611391

14621392
fn next(&mut self) -> Option<&'a T> {
14631393
loop {
1464-
let (a_next, b_next) = self.0.nexts();
1394+
let (a_next, b_next) = self.0.nexts(Self::Item::cmp);
14651395
if a_next.and(b_next).is_none() {
14661396
return a_next.or(b_next);
14671397
}
@@ -1555,7 +1485,7 @@ impl<'a, T: Ord> Iterator for Union<'a, T> {
15551485
type Item = &'a T;
15561486

15571487
fn next(&mut self) -> Option<&'a T> {
1558-
let (a_next, b_next) = self.0.nexts();
1488+
let (a_next, b_next) = self.0.nexts(Self::Item::cmp);
15591489
a_next.or(b_next)
15601490
}
15611491

0 commit comments

Comments
 (0)