Skip to content

Commit 4eb069c

Browse files
committed
Don't allocate during default HashSet creation.
The following `HashMap` creation functions don't allocate heap storage for elements. ``` HashMap::new() HashMap::default() HashMap::with_hasher() ``` This is good, because it's surprisingly common to create a HashMap and never use it. So that case should be cheap. However, `HashSet` does not have the same behaviour. The corresponding creation functions *do* allocate heap storage for the default number of non-zero elements (which is 32 slots for 29 elements). ``` HashMap::new() HashMap::default() HashMap::with_hasher() ``` This commit gives `HashSet` the same behaviour as `HashMap`, by simply calling the corresponding `HashMap` functions (something `HashSet` already does for `with_capacity` and `with_capacity_and_hasher`). It also reformats one existing `HashSet` construction to use a consistent single-line format. This speeds up rustc itself by 1.01--1.04x on most of the non-tiny rustc-benchmarks.
1 parent 458f411 commit 4eb069c

File tree

2 files changed

+34
-8
lines changed

2 files changed

+34
-8
lines changed

src/libstd/collections/hash/map.rs

+15
Original file line numberDiff line numberDiff line change
@@ -2087,9 +2087,24 @@ fn assert_covariance() {
20872087
mod test_map {
20882088
use super::HashMap;
20892089
use super::Entry::{Occupied, Vacant};
2090+
use super::RandomState;
20902091
use cell::RefCell;
20912092
use rand::{thread_rng, Rng};
20922093

2094+
#[test]
2095+
fn test_create_capacities() {
2096+
type HM = HashMap<i32, i32>;
2097+
2098+
let m = HM::new();
2099+
assert_eq!(m.capacity(), 0);
2100+
2101+
let m = HM::default();
2102+
assert_eq!(m.capacity(), 0);
2103+
2104+
let m = HM::with_hasher(RandomState::new());
2105+
assert_eq!(m.capacity(), 0);
2106+
}
2107+
20932108
#[test]
20942109
fn test_create_capacity_zero() {
20952110
let mut m = HashMap::with_capacity(0);

src/libstd/collections/hash/set.rs

+19-8
Original file line numberDiff line numberDiff line change
@@ -17,8 +17,6 @@ use ops::{BitOr, BitAnd, BitXor, Sub};
1717
use super::Recover;
1818
use super::map::{self, HashMap, Keys, RandomState};
1919

20-
const INITIAL_CAPACITY: usize = 32;
21-
2220
// Future Optimization (FIXME!)
2321
// =============================
2422
//
@@ -118,7 +116,7 @@ impl<T: Hash + Eq> HashSet<T, RandomState> {
118116
#[inline]
119117
#[stable(feature = "rust1", since = "1.0.0")]
120118
pub fn new() -> HashSet<T, RandomState> {
121-
HashSet::with_capacity(INITIAL_CAPACITY)
119+
HashSet { map: HashMap::new() }
122120
}
123121

124122
/// Creates an empty HashSet with space for at least `n` elements in
@@ -163,7 +161,7 @@ impl<T, S> HashSet<T, S>
163161
#[inline]
164162
#[stable(feature = "hashmap_build_hasher", since = "1.7.0")]
165163
pub fn with_hasher(hasher: S) -> HashSet<T, S> {
166-
HashSet::with_capacity_and_hasher(INITIAL_CAPACITY, hasher)
164+
HashSet { map: HashMap::with_hasher(hasher) }
167165
}
168166

169167
/// Creates an empty HashSet with space for at least `capacity`
@@ -188,9 +186,7 @@ impl<T, S> HashSet<T, S>
188186
#[stable(feature = "hashmap_build_hasher", since = "1.7.0")]
189187
pub fn with_capacity_and_hasher(capacity: usize, hasher: S)
190188
-> HashSet<T, S> {
191-
HashSet {
192-
map: HashMap::with_capacity_and_hasher(capacity, hasher),
193-
}
189+
HashSet { map: HashMap::with_capacity_and_hasher(capacity, hasher) }
194190
}
195191

196192
/// Returns a reference to the set's hasher.
@@ -667,7 +663,7 @@ impl<T, S> Default for HashSet<T, S>
667663
{
668664
/// Creates an empty `HashSet<T, S>` with the `Default` value for the hasher.
669665
fn default() -> HashSet<T, S> {
670-
HashSet::with_hasher(Default::default())
666+
HashSet { map: HashMap::default() }
671667
}
672668
}
673669

@@ -1069,6 +1065,21 @@ fn assert_covariance() {
10691065
#[cfg(test)]
10701066
mod test_set {
10711067
use super::HashSet;
1068+
use super::super::map::RandomState;
1069+
1070+
#[test]
1071+
fn test_create_capacities() {
1072+
type HS = HashSet<i32>;
1073+
1074+
let s = HS::new();
1075+
assert_eq!(s.capacity(), 0);
1076+
1077+
let s = HS::default();
1078+
assert_eq!(s.capacity(), 0);
1079+
1080+
let s = HS::with_hasher(RandomState::new());
1081+
assert_eq!(s.capacity(), 0);
1082+
}
10721083

10731084
#[test]
10741085
fn test_disjoint() {

0 commit comments

Comments
 (0)