Skip to content

Commit 009969a

Browse files
committed
Auto merge of #374 - the8472:dont-drain-on-drop, r=Amanieu
Remove drain-on-drop behavior from DrainFilter This is a breaking change for hashbrown but not for std because in std the drain impl is still unstable. This is part of stdlib [ACP 136](rust-lang/libs-team#136).
2 parents 329f86a + a0ab1de commit 009969a

File tree

2 files changed

+48
-91
lines changed

2 files changed

+48
-91
lines changed

src/map.rs

+30-57
Original file line numberDiff line numberDiff line change
@@ -972,15 +972,12 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
972972
/// In other words, move all pairs `(k, v)` such that `f(&k, &mut v)` returns `true` out
973973
/// into another iterator.
974974
///
975-
/// Note that `drain_filter` lets you mutate every value in the filter closure, regardless of
975+
/// Note that `extract_if` lets you mutate every value in the filter closure, regardless of
976976
/// whether you choose to keep or remove it.
977977
///
978-
/// When the returned DrainedFilter is dropped, any remaining elements that satisfy
979-
/// the predicate are dropped from the table.
980-
///
981-
/// It is unspecified how many more elements will be subjected to the closure
982-
/// if a panic occurs in the closure, or a panic occurs while dropping an element,
983-
/// or if the `DrainFilter` value is leaked.
978+
/// If the returned `ExtractIf` is not exhausted, e.g. because it is dropped without iterating
979+
/// or the iteration short-circuits, then the remaining elements will be retained.
980+
/// Use [`retain()`] with a negated predicate if you do not need the returned iterator.
984981
///
985982
/// Keeps the allocated memory for reuse.
986983
///
@@ -991,7 +988,7 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
991988
///
992989
/// let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x)).collect();
993990
///
994-
/// let drained: HashMap<i32, i32> = map.drain_filter(|k, _v| k % 2 == 0).collect();
991+
/// let drained: HashMap<i32, i32> = map.extract_if(|k, _v| k % 2 == 0).collect();
995992
///
996993
/// let mut evens = drained.keys().cloned().collect::<Vec<_>>();
997994
/// let mut odds = map.keys().cloned().collect::<Vec<_>>();
@@ -1004,21 +1001,20 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
10041001
/// let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x)).collect();
10051002
///
10061003
/// { // Iterator is dropped without being consumed.
1007-
/// let d = map.drain_filter(|k, _v| k % 2 != 0);
1004+
/// let d = map.extract_if(|k, _v| k % 2 != 0);
10081005
/// }
10091006
///
1010-
/// // But the map lens have been reduced by half
1011-
/// // even if we do not use DrainFilter iterator.
1012-
/// assert_eq!(map.len(), 4);
1007+
/// // ExtractIf was not exhausted, therefore no elements were drained.
1008+
/// assert_eq!(map.len(), 8);
10131009
/// ```
10141010
#[cfg_attr(feature = "inline-more", inline)]
1015-
pub fn drain_filter<F>(&mut self, f: F) -> DrainFilter<'_, K, V, F, A>
1011+
pub fn extract_if<F>(&mut self, f: F) -> ExtractIf<'_, K, V, F, A>
10161012
where
10171013
F: FnMut(&K, &mut V) -> bool,
10181014
{
1019-
DrainFilter {
1015+
ExtractIf {
10201016
f,
1021-
inner: DrainFilterInner {
1017+
inner: ExtractIfInner {
10221018
iter: unsafe { self.table.iter() },
10231019
table: &mut self.table,
10241020
},
@@ -2732,10 +2728,10 @@ impl<K, V, A: Allocator + Clone> Drain<'_, K, V, A> {
27322728
/// A draining iterator over entries of a `HashMap` which don't satisfy the predicate
27332729
/// `f(&k, &mut v)` in arbitrary order. The iterator element type is `(K, V)`.
27342730
///
2735-
/// This `struct` is created by the [`drain_filter`] method on [`HashMap`]. See its
2731+
/// This `struct` is created by the [`extract_if`] method on [`HashMap`]. See its
27362732
/// documentation for more.
27372733
///
2738-
/// [`drain_filter`]: struct.HashMap.html#method.drain_filter
2734+
/// [`extract_if`]: struct.HashMap.html#method.extract_if
27392735
/// [`HashMap`]: struct.HashMap.html
27402736
///
27412737
/// # Examples
@@ -2745,54 +2741,31 @@ impl<K, V, A: Allocator + Clone> Drain<'_, K, V, A> {
27452741
///
27462742
/// let mut map: HashMap<i32, &str> = [(1, "a"), (2, "b"), (3, "c")].into();
27472743
///
2748-
/// let mut drain_filter = map.drain_filter(|k, _v| k % 2 != 0);
2749-
/// let mut vec = vec![drain_filter.next(), drain_filter.next()];
2744+
/// let mut extract_if = map.extract_if(|k, _v| k % 2 != 0);
2745+
/// let mut vec = vec![extract_if.next(), extract_if.next()];
27502746
///
2751-
/// // The `DrainFilter` iterator produces items in arbitrary order, so the
2747+
/// // The `ExtractIf` iterator produces items in arbitrary order, so the
27522748
/// // items must be sorted to test them against a sorted array.
27532749
/// vec.sort_unstable();
27542750
/// assert_eq!(vec, [Some((1, "a")),Some((3, "c"))]);
27552751
///
27562752
/// // It is fused iterator
2757-
/// assert_eq!(drain_filter.next(), None);
2758-
/// assert_eq!(drain_filter.next(), None);
2759-
/// drop(drain_filter);
2753+
/// assert_eq!(extract_if.next(), None);
2754+
/// assert_eq!(extract_if.next(), None);
2755+
/// drop(extract_if);
27602756
///
27612757
/// assert_eq!(map.len(), 1);
27622758
/// ```
2763-
pub struct DrainFilter<'a, K, V, F, A: Allocator + Clone = Global>
2759+
#[must_use = "Iterators are lazy unless consumed"]
2760+
pub struct ExtractIf<'a, K, V, F, A: Allocator + Clone = Global>
27642761
where
27652762
F: FnMut(&K, &mut V) -> bool,
27662763
{
27672764
f: F,
2768-
inner: DrainFilterInner<'a, K, V, A>,
2769-
}
2770-
2771-
impl<'a, K, V, F, A> Drop for DrainFilter<'a, K, V, F, A>
2772-
where
2773-
F: FnMut(&K, &mut V) -> bool,
2774-
A: Allocator + Clone,
2775-
{
2776-
#[cfg_attr(feature = "inline-more", inline)]
2777-
fn drop(&mut self) {
2778-
while let Some(item) = self.next() {
2779-
let guard = ConsumeAllOnDrop(self);
2780-
drop(item);
2781-
mem::forget(guard);
2782-
}
2783-
}
2784-
}
2785-
2786-
pub(super) struct ConsumeAllOnDrop<'a, T: Iterator>(pub &'a mut T);
2787-
2788-
impl<T: Iterator> Drop for ConsumeAllOnDrop<'_, T> {
2789-
#[cfg_attr(feature = "inline-more", inline)]
2790-
fn drop(&mut self) {
2791-
self.0.for_each(drop);
2792-
}
2765+
inner: ExtractIfInner<'a, K, V, A>,
27932766
}
27942767

2795-
impl<K, V, F, A> Iterator for DrainFilter<'_, K, V, F, A>
2768+
impl<K, V, F, A> Iterator for ExtractIf<'_, K, V, F, A>
27962769
where
27972770
F: FnMut(&K, &mut V) -> bool,
27982771
A: Allocator + Clone,
@@ -2810,15 +2783,15 @@ where
28102783
}
28112784
}
28122785

2813-
impl<K, V, F> FusedIterator for DrainFilter<'_, K, V, F> where F: FnMut(&K, &mut V) -> bool {}
2786+
impl<K, V, F> FusedIterator for ExtractIf<'_, K, V, F> where F: FnMut(&K, &mut V) -> bool {}
28142787

2815-
/// Portions of `DrainFilter` shared with `set::DrainFilter`
2816-
pub(super) struct DrainFilterInner<'a, K, V, A: Allocator + Clone> {
2788+
/// Portions of `ExtractIf` shared with `set::ExtractIf`
2789+
pub(super) struct ExtractIfInner<'a, K, V, A: Allocator + Clone> {
28172790
pub iter: RawIter<(K, V)>,
28182791
pub table: &'a mut RawTable<(K, V), A>,
28192792
}
28202793

2821-
impl<K, V, A: Allocator + Clone> DrainFilterInner<'_, K, V, A> {
2794+
impl<K, V, A: Allocator + Clone> ExtractIfInner<'_, K, V, A> {
28222795
#[cfg_attr(feature = "inline-more", inline)]
28232796
pub(super) fn next<F>(&mut self, f: &mut F) -> Option<(K, V)>
28242797
where
@@ -8169,18 +8142,18 @@ mod test_map {
81698142
}
81708143

81718144
#[test]
8172-
fn test_drain_filter() {
8145+
fn test_extract_if() {
81738146
{
81748147
let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x * 10)).collect();
8175-
let drained = map.drain_filter(|&k, _| k % 2 == 0);
8148+
let drained = map.extract_if(|&k, _| k % 2 == 0);
81768149
let mut out = drained.collect::<Vec<_>>();
81778150
out.sort_unstable();
81788151
assert_eq!(vec![(0, 0), (2, 20), (4, 40), (6, 60)], out);
81798152
assert_eq!(map.len(), 4);
81808153
}
81818154
{
81828155
let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x * 10)).collect();
8183-
drop(map.drain_filter(|&k, _| k % 2 == 0));
8156+
map.extract_if(|&k, _| k % 2 == 0).for_each(drop);
81848157
assert_eq!(map.len(), 4);
81858158
}
81868159
}

src/set.rs

+18-34
Original file line numberDiff line numberDiff line change
@@ -5,10 +5,9 @@ use alloc::borrow::ToOwned;
55
use core::fmt;
66
use core::hash::{BuildHasher, Hash};
77
use core::iter::{Chain, FromIterator, FusedIterator};
8-
use core::mem;
98
use core::ops::{BitAnd, BitOr, BitXor, Sub};
109

11-
use super::map::{self, ConsumeAllOnDrop, DefaultHashBuilder, DrainFilterInner, HashMap, Keys};
10+
use super::map::{self, DefaultHashBuilder, ExtractIfInner, HashMap, Keys};
1211
use crate::raw::{Allocator, Global};
1312

1413
// Future Optimization (FIXME!)
@@ -380,16 +379,17 @@ impl<T, S, A: Allocator + Clone> HashSet<T, S, A> {
380379
/// In other words, move all elements `e` such that `f(&e)` returns `true` out
381380
/// into another iterator.
382381
///
383-
/// When the returned DrainedFilter is dropped, any remaining elements that satisfy
384-
/// the predicate are dropped from the set.
382+
/// If the returned `ExtractIf` is not exhausted, e.g. because it is dropped without iterating
383+
/// or the iteration short-circuits, then the remaining elements will be retained.
384+
/// Use [`retain()`] with a negated predicate if you do not need the returned iterator.
385385
///
386386
/// # Examples
387387
///
388388
/// ```
389389
/// use hashbrown::HashSet;
390390
///
391391
/// let mut set: HashSet<i32> = (0..8).collect();
392-
/// let drained: HashSet<i32> = set.drain_filter(|v| v % 2 == 0).collect();
392+
/// let drained: HashSet<i32> = set.extract_if(|v| v % 2 == 0).collect();
393393
///
394394
/// let mut evens = drained.into_iter().collect::<Vec<_>>();
395395
/// let mut odds = set.into_iter().collect::<Vec<_>>();
@@ -400,13 +400,13 @@ impl<T, S, A: Allocator + Clone> HashSet<T, S, A> {
400400
/// assert_eq!(odds, vec![1, 3, 5, 7]);
401401
/// ```
402402
#[cfg_attr(feature = "inline-more", inline)]
403-
pub fn drain_filter<F>(&mut self, f: F) -> DrainFilter<'_, T, F, A>
403+
pub fn extract_if<F>(&mut self, f: F) -> ExtractIf<'_, T, F, A>
404404
where
405405
F: FnMut(&T) -> bool,
406406
{
407-
DrainFilter {
407+
ExtractIf {
408408
f,
409-
inner: DrainFilterInner {
409+
inner: ExtractIfInner {
410410
iter: unsafe { self.map.table.iter() },
411411
table: &mut self.map.table,
412412
},
@@ -1567,17 +1567,18 @@ pub struct Drain<'a, K, A: Allocator + Clone = Global> {
15671567

15681568
/// A draining iterator over entries of a `HashSet` which don't satisfy the predicate `f`.
15691569
///
1570-
/// This `struct` is created by the [`drain_filter`] method on [`HashSet`]. See its
1570+
/// This `struct` is created by the [`extract_if`] method on [`HashSet`]. See its
15711571
/// documentation for more.
15721572
///
1573-
/// [`drain_filter`]: struct.HashSet.html#method.drain_filter
1573+
/// [`extract_if`]: struct.HashSet.html#method.extract_if
15741574
/// [`HashSet`]: struct.HashSet.html
1575-
pub struct DrainFilter<'a, K, F, A: Allocator + Clone = Global>
1575+
#[must_use = "Iterators are lazy unless consumed"]
1576+
pub struct ExtractIf<'a, K, F, A: Allocator + Clone = Global>
15761577
where
15771578
F: FnMut(&K) -> bool,
15781579
{
15791580
f: F,
1580-
inner: DrainFilterInner<'a, K, (), A>,
1581+
inner: ExtractIfInner<'a, K, (), A>,
15811582
}
15821583

15831584
/// A lazy iterator producing elements in the intersection of `HashSet`s.
@@ -1768,21 +1769,7 @@ impl<K: fmt::Debug, A: Allocator + Clone> fmt::Debug for Drain<'_, K, A> {
17681769
}
17691770
}
17701771

1771-
impl<'a, K, F, A: Allocator + Clone> Drop for DrainFilter<'a, K, F, A>
1772-
where
1773-
F: FnMut(&K) -> bool,
1774-
{
1775-
#[cfg_attr(feature = "inline-more", inline)]
1776-
fn drop(&mut self) {
1777-
while let Some(item) = self.next() {
1778-
let guard = ConsumeAllOnDrop(self);
1779-
drop(item);
1780-
mem::forget(guard);
1781-
}
1782-
}
1783-
}
1784-
1785-
impl<K, F, A: Allocator + Clone> Iterator for DrainFilter<'_, K, F, A>
1772+
impl<K, F, A: Allocator + Clone> Iterator for ExtractIf<'_, K, F, A>
17861773
where
17871774
F: FnMut(&K) -> bool,
17881775
{
@@ -1801,10 +1788,7 @@ where
18011788
}
18021789
}
18031790

1804-
impl<K, F, A: Allocator + Clone> FusedIterator for DrainFilter<'_, K, F, A> where
1805-
F: FnMut(&K) -> bool
1806-
{
1807-
}
1791+
impl<K, F, A: Allocator + Clone> FusedIterator for ExtractIf<'_, K, F, A> where F: FnMut(&K) -> bool {}
18081792

18091793
impl<T, S, A: Allocator + Clone> Clone for Intersection<'_, T, S, A> {
18101794
#[cfg_attr(feature = "inline-more", inline)]
@@ -2850,18 +2834,18 @@ mod test_set {
28502834
}
28512835

28522836
#[test]
2853-
fn test_drain_filter() {
2837+
fn test_extract_if() {
28542838
{
28552839
let mut set: HashSet<i32> = (0..8).collect();
2856-
let drained = set.drain_filter(|&k| k % 2 == 0);
2840+
let drained = set.extract_if(|&k| k % 2 == 0);
28572841
let mut out = drained.collect::<Vec<_>>();
28582842
out.sort_unstable();
28592843
assert_eq!(vec![0, 2, 4, 6], out);
28602844
assert_eq!(set.len(), 4);
28612845
}
28622846
{
28632847
let mut set: HashSet<i32> = (0..8).collect();
2864-
drop(set.drain_filter(|&k| k % 2 == 0));
2848+
set.extract_if(|&k| k % 2 == 0).for_each(drop);
28652849
assert_eq!(set.len(), 4, "Removes non-matching items on drop");
28662850
}
28672851
}

0 commit comments

Comments
 (0)