1
1
use crate :: raw:: { Allocator , Bucket , Global , RawDrain , RawIntoIter , RawIter , RawTable } ;
2
2
use crate :: TryReserveError ;
3
+ #[ cfg( feature = "nightly" ) ]
4
+ use crate :: UnavailableMutError ;
3
5
use core:: borrow:: Borrow ;
4
6
use core:: fmt:: { self , Debug } ;
5
7
use core:: hash:: { BuildHasher , Hash } ;
6
8
use core:: iter:: { FromIterator , FusedIterator } ;
7
9
use core:: marker:: PhantomData ;
8
10
use core:: mem;
11
+ #[ cfg( feature = "nightly" ) ]
12
+ use core:: mem:: MaybeUninit ;
9
13
use core:: ops:: Index ;
10
14
11
15
/// Default hasher for `HashMap`.
@@ -1113,6 +1117,137 @@ where
1113
1117
self . table . get_mut ( hash, equivalent_key ( k) )
1114
1118
}
1115
1119
1120
+ /// Attempts to get mutable references to `N` values in the map at once.
1121
+ ///
1122
+ /// Returns an array of length `N` with the results of each query. For soundness,
1123
+ /// at most one mutable reference will be returned to any value. An
1124
+ /// `Err(UnavailableMutError::Duplicate(i))` in the returned array indicates that a suitable
1125
+ /// key-value pair exists, but a mutable reference to the value already occurs at index `i` in
1126
+ /// the returned array.
1127
+ ///
1128
+ /// This method is available only if the `nightly` feature is enabled.
1129
+ ///
1130
+ /// ```
1131
+ /// use hashbrown::{HashMap, UnavailableMutError};
1132
+ ///
1133
+ /// let mut libraries = HashMap::new();
1134
+ /// libraries.insert("Bodleian Library".to_string(), 1602);
1135
+ /// libraries.insert("Athenæum".to_string(), 1807);
1136
+ /// libraries.insert("Herzogin-Anna-Amalia-Bibliothek".to_string(), 1691);
1137
+ /// libraries.insert("Library of Congress".to_string(), 1800);
1138
+ ///
1139
+ /// let got = libraries.get_each_mut([
1140
+ /// "Athenæum",
1141
+ /// "New York Public Library",
1142
+ /// "Athenæum",
1143
+ /// "Library of Congress",
1144
+ /// ]);
1145
+ /// assert_eq!(
1146
+ /// got,
1147
+ /// [
1148
+ /// Ok(&mut 1807),
1149
+ /// Err(UnavailableMutError::Absent),
1150
+ /// Err(UnavailableMutError::Duplicate(0)),
1151
+ /// Ok(&mut 1800),
1152
+ /// ]
1153
+ /// );
1154
+ /// ```
1155
+ #[ cfg( feature = "nightly" ) ]
1156
+ pub fn get_each_mut < Q : ?Sized , const N : usize > (
1157
+ & mut self ,
1158
+ ks : [ & Q ; N ] ,
1159
+ ) -> [ Result < & ' _ mut V , UnavailableMutError > ; N ]
1160
+ where
1161
+ K : Borrow < Q > ,
1162
+ Q : Hash + Eq ,
1163
+ {
1164
+ let mut pairs = self . get_each_inner_mut ( ks) ;
1165
+ // TODO use `MaybeUninit::uninit_array` here instead once that's stable.
1166
+ let mut out: [ MaybeUninit < Result < & ' _ mut V , UnavailableMutError > > ; N ] =
1167
+ unsafe { MaybeUninit :: uninit ( ) . assume_init ( ) } ;
1168
+ for i in 0 ..N {
1169
+ out[ i] = MaybeUninit :: new (
1170
+ mem:: replace ( & mut pairs[ i] , Err ( UnavailableMutError :: Absent ) ) . map ( |( _, v) | v) ,
1171
+ ) ;
1172
+ }
1173
+ unsafe { MaybeUninit :: array_assume_init ( out) }
1174
+ }
1175
+
1176
+ /// Attempts to get mutable references to `N` values in the map at once, with immutable
1177
+ /// references to the corresponding keys.
1178
+ ///
1179
+ /// Returns an array of length `N` with the results of each query. For soundness,
1180
+ /// at most one mutable reference will be returned to any value. An
1181
+ /// `Err(UnavailableMutError::Duplicate(i))` in the returned array indicates that a suitable
1182
+ /// key-value pair exists, but a mutable reference to the value already occurs at index `i` in
1183
+ /// the returned array.
1184
+ ///
1185
+ /// This method is available only if the `nightly` feature is enabled.
1186
+ ///
1187
+ /// ```
1188
+ /// use hashbrown::{HashMap, UnavailableMutError};
1189
+ ///
1190
+ /// let mut libraries = HashMap::new();
1191
+ /// libraries.insert("Bodleian Library".to_string(), 1602);
1192
+ /// libraries.insert("Athenæum".to_string(), 1807);
1193
+ /// libraries.insert("Herzogin-Anna-Amalia-Bibliothek".to_string(), 1691);
1194
+ /// libraries.insert("Library of Congress".to_string(), 1800);
1195
+ ///
1196
+ /// let got = libraries.get_each_key_value_mut([
1197
+ /// "Bodleian Library",
1198
+ /// "Herzogin-Anna-Amalia-Bibliothek",
1199
+ /// "Herzogin-Anna-Amalia-Bibliothek",
1200
+ /// "Gewandhaus",
1201
+ /// ]);
1202
+ /// assert_eq!(
1203
+ /// got,
1204
+ /// [
1205
+ /// Ok((&"Bodleian Library".to_string(), &mut 1602)),
1206
+ /// Ok((&"Herzogin-Anna-Amalia-Bibliothek".to_string(), &mut 1691)),
1207
+ /// Err(UnavailableMutError::Duplicate(1)),
1208
+ /// Err(UnavailableMutError::Absent),
1209
+ /// ]
1210
+ /// );
1211
+ /// ```
1212
+ #[ cfg( feature = "nightly" ) ]
1213
+ pub fn get_each_key_value_mut < Q : ?Sized , const N : usize > (
1214
+ & mut self ,
1215
+ ks : [ & Q ; N ] ,
1216
+ ) -> [ Result < ( & ' _ K , & ' _ mut V ) , UnavailableMutError > ; N ]
1217
+ where
1218
+ K : Borrow < Q > ,
1219
+ Q : Hash + Eq ,
1220
+ {
1221
+ let mut pairs = self . get_each_inner_mut ( ks) ;
1222
+ // TODO use `MaybeUninit::uninit_array` here instead once that's stable.
1223
+ let mut out: [ MaybeUninit < Result < ( & ' _ K , & ' _ mut V ) , UnavailableMutError > > ; N ] =
1224
+ unsafe { MaybeUninit :: uninit ( ) . assume_init ( ) } ;
1225
+ for i in 0 ..N {
1226
+ out[ i] = MaybeUninit :: new (
1227
+ mem:: replace ( & mut pairs[ i] , Err ( UnavailableMutError :: Absent ) )
1228
+ . map ( |( k, v) | ( & * k, v) ) ,
1229
+ ) ;
1230
+ }
1231
+ unsafe { MaybeUninit :: array_assume_init ( out) }
1232
+ }
1233
+
1234
+ #[ cfg( feature = "nightly" ) ]
1235
+ fn get_each_inner_mut < Q : ?Sized , const N : usize > (
1236
+ & mut self ,
1237
+ ks : [ & Q ; N ] ,
1238
+ ) -> [ Result < & ' _ mut ( K , V ) , UnavailableMutError > ; N ]
1239
+ where
1240
+ K : Borrow < Q > ,
1241
+ Q : Hash + Eq ,
1242
+ {
1243
+ let mut hashes = [ 0_u64 ; N ] ;
1244
+ for i in 0 ..N {
1245
+ hashes[ i] = make_hash :: < K , Q , S > ( & self . hash_builder , ks[ i] ) ;
1246
+ }
1247
+ self . table
1248
+ . get_each_mut ( hashes, |i, ( k, _) | ks[ i] . eq ( k. borrow ( ) ) )
1249
+ }
1250
+
1116
1251
/// Inserts a key-value pair into the map.
1117
1252
///
1118
1253
/// If the map did not have this key present, [`None`] is returned.
@@ -3315,6 +3450,7 @@ mod test_map {
3315
3450
use super :: { HashMap , RawEntryMut } ;
3316
3451
use crate :: TryReserveError :: * ;
3317
3452
use rand:: { rngs:: SmallRng , Rng , SeedableRng } ;
3453
+ use std:: borrow:: ToOwned ;
3318
3454
use std:: cell:: RefCell ;
3319
3455
use std:: usize;
3320
3456
use std:: vec:: Vec ;
@@ -4682,7 +4818,6 @@ mod test_map {
4682
4818
#[ test]
4683
4819
fn test_const_with_hasher ( ) {
4684
4820
use core:: hash:: BuildHasher ;
4685
- use std:: borrow:: ToOwned ;
4686
4821
use std:: collections:: hash_map:: DefaultHasher ;
4687
4822
4688
4823
#[ derive( Clone ) ]
@@ -4702,4 +4837,33 @@ mod test_map {
4702
4837
map. insert ( 17 , "seventeen" . to_owned ( ) ) ;
4703
4838
assert_eq ! ( "seventeen" , map[ & 17 ] ) ;
4704
4839
}
4840
+
4841
+ #[ test]
4842
+ #[ cfg( feature = "nightly" ) ]
4843
+ fn test_get_each_mut ( ) {
4844
+ use crate :: UnavailableMutError :: * ;
4845
+
4846
+ let mut map = HashMap :: new ( ) ;
4847
+ map. insert ( "foo" . to_owned ( ) , 0 ) ;
4848
+ map. insert ( "bar" . to_owned ( ) , 10 ) ;
4849
+ map. insert ( "baz" . to_owned ( ) , 20 ) ;
4850
+ map. insert ( "qux" . to_owned ( ) , 30 ) ;
4851
+
4852
+ let xs = map. get_each_mut ( [ "foo" , "dud" , "foo" , "qux" ] ) ;
4853
+ assert_eq ! (
4854
+ xs,
4855
+ [ Ok ( & mut 0 ) , Err ( Absent ) , Err ( Duplicate ( 0 ) ) , Ok ( & mut 30 ) ]
4856
+ ) ;
4857
+
4858
+ let ys = map. get_each_key_value_mut ( [ "bar" , "baz" , "baz" , "dip" ] ) ;
4859
+ assert_eq ! (
4860
+ ys,
4861
+ [
4862
+ Ok ( ( & "bar" . to_owned( ) , & mut 10 ) ) ,
4863
+ Ok ( ( & "baz" . to_owned( ) , & mut 20 ) ) ,
4864
+ Err ( Duplicate ( 1 ) ) ,
4865
+ Err ( Absent ) ,
4866
+ ]
4867
+ ) ;
4868
+ }
4705
4869
}
0 commit comments