@@ -972,15 +972,12 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
972
972
/// In other words, move all pairs `(k, v)` such that `f(&k, &mut v)` returns `true` out
973
973
/// into another iterator.
974
974
///
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
976
976
/// whether you choose to keep or remove it.
977
977
///
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.
984
981
///
985
982
/// Keeps the allocated memory for reuse.
986
983
///
@@ -991,7 +988,7 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
991
988
///
992
989
/// let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x)).collect();
993
990
///
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();
995
992
///
996
993
/// let mut evens = drained.keys().cloned().collect::<Vec<_>>();
997
994
/// let mut odds = map.keys().cloned().collect::<Vec<_>>();
@@ -1004,21 +1001,20 @@ impl<K, V, S, A: Allocator + Clone> HashMap<K, V, S, A> {
1004
1001
/// let mut map: HashMap<i32, i32> = (0..8).map(|x| (x, x)).collect();
1005
1002
///
1006
1003
/// { // 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);
1008
1005
/// }
1009
1006
///
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);
1013
1009
/// ```
1014
1010
#[ 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 >
1016
1012
where
1017
1013
F : FnMut ( & K , & mut V ) -> bool ,
1018
1014
{
1019
- DrainFilter {
1015
+ ExtractIf {
1020
1016
f,
1021
- inner : DrainFilterInner {
1017
+ inner : ExtractIfInner {
1022
1018
iter : unsafe { self . table . iter ( ) } ,
1023
1019
table : & mut self . table ,
1024
1020
} ,
@@ -2732,10 +2728,10 @@ impl<K, V, A: Allocator + Clone> Drain<'_, K, V, A> {
2732
2728
/// A draining iterator over entries of a `HashMap` which don't satisfy the predicate
2733
2729
/// `f(&k, &mut v)` in arbitrary order. The iterator element type is `(K, V)`.
2734
2730
///
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
2736
2732
/// documentation for more.
2737
2733
///
2738
- /// [`drain_filter `]: struct.HashMap.html#method.drain_filter
2734
+ /// [`extract_if `]: struct.HashMap.html#method.extract_if
2739
2735
/// [`HashMap`]: struct.HashMap.html
2740
2736
///
2741
2737
/// # Examples
@@ -2745,54 +2741,31 @@ impl<K, V, A: Allocator + Clone> Drain<'_, K, V, A> {
2745
2741
///
2746
2742
/// let mut map: HashMap<i32, &str> = [(1, "a"), (2, "b"), (3, "c")].into();
2747
2743
///
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()];
2750
2746
///
2751
- /// // The `DrainFilter ` iterator produces items in arbitrary order, so the
2747
+ /// // The `ExtractIf ` iterator produces items in arbitrary order, so the
2752
2748
/// // items must be sorted to test them against a sorted array.
2753
2749
/// vec.sort_unstable();
2754
2750
/// assert_eq!(vec, [Some((1, "a")),Some((3, "c"))]);
2755
2751
///
2756
2752
/// // 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 );
2760
2756
///
2761
2757
/// assert_eq!(map.len(), 1);
2762
2758
/// ```
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 >
2764
2761
where
2765
2762
F : FnMut ( & K , & mut V ) -> bool ,
2766
2763
{
2767
2764
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 > ,
2793
2766
}
2794
2767
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 >
2796
2769
where
2797
2770
F : FnMut ( & K , & mut V ) -> bool ,
2798
2771
A : Allocator + Clone ,
@@ -2810,15 +2783,15 @@ where
2810
2783
}
2811
2784
}
2812
2785
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 { }
2814
2787
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 > {
2817
2790
pub iter : RawIter < ( K , V ) > ,
2818
2791
pub table : & ' a mut RawTable < ( K , V ) , A > ,
2819
2792
}
2820
2793
2821
- impl < K , V , A : Allocator + Clone > DrainFilterInner < ' _ , K , V , A > {
2794
+ impl < K , V , A : Allocator + Clone > ExtractIfInner < ' _ , K , V , A > {
2822
2795
#[ cfg_attr( feature = "inline-more" , inline) ]
2823
2796
pub ( super ) fn next < F > ( & mut self , f : & mut F ) -> Option < ( K , V ) >
2824
2797
where
@@ -8169,18 +8142,18 @@ mod test_map {
8169
8142
}
8170
8143
8171
8144
#[ test]
8172
- fn test_drain_filter ( ) {
8145
+ fn test_extract_if ( ) {
8173
8146
{
8174
8147
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 ) ;
8176
8149
let mut out = drained. collect :: < Vec < _ > > ( ) ;
8177
8150
out. sort_unstable ( ) ;
8178
8151
assert_eq ! ( vec![ ( 0 , 0 ) , ( 2 , 20 ) , ( 4 , 40 ) , ( 6 , 60 ) ] , out) ;
8179
8152
assert_eq ! ( map. len( ) , 4 ) ;
8180
8153
}
8181
8154
{
8182
8155
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 ) ;
8184
8157
assert_eq ! ( map. len( ) , 4 ) ;
8185
8158
}
8186
8159
}
0 commit comments