2014-02-28 22:23:53 -05:00
|
|
|
|
// Copyright 2014 The Rust Project Developers. See the COPYRIGHT
|
2012-12-03 16:48:01 -08:00
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
|
//
|
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
|
// except according to those terms.
|
2014-07-21 15:57:14 -07:00
|
|
|
|
//
|
|
|
|
|
// ignore-lexer-test FIXME #15883
|
2012-12-03 16:48:01 -08:00
|
|
|
|
|
2014-12-13 11:15:18 -05:00
|
|
|
|
use self::Entry::*;
|
2014-11-06 00:05:53 -08:00
|
|
|
|
use self::SearchResult::*;
|
|
|
|
|
use self::VacantEntryState::*;
|
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
use borrow::BorrowFrom;
|
std: Recreate a `collections` module
As with the previous commit with `librand`, this commit shuffles around some
`collections` code. The new state of the world is similar to that of librand:
* The libcollections crate now only depends on libcore and liballoc.
* The standard library has a new module, `std::collections`. All functionality
of libcollections is reexported through this module.
I would like to stress that this change is purely cosmetic. There are very few
alterations to these primitives.
There are a number of notable points about the new organization:
* std::{str, slice, string, vec} all moved to libcollections. There is no reason
that these primitives shouldn't be necessarily usable in a freestanding
context that has allocation. These are all reexported in their usual places in
the standard library.
* The `hashmap`, and transitively the `lru_cache`, modules no longer reside in
`libcollections`, but rather in libstd. The reason for this is because the
`HashMap::new` contructor requires access to the OSRng for initially seeding
the hash map. Beyond this requirement, there is no reason that the hashmap
could not move to libcollections.
I do, however, have a plan to move the hash map to the collections module. The
`HashMap::new` function could be altered to require that the `H` hasher
parameter ascribe to the `Default` trait, allowing the entire `hashmap` module
to live in libcollections. The key idea would be that the default hasher would
be different in libstd. Something along the lines of:
// src/libstd/collections/mod.rs
pub type HashMap<K, V, H = RandomizedSipHasher> =
core_collections::HashMap<K, V, H>;
This is not possible today because you cannot invoke static methods through
type aliases. If we modified the compiler, however, to allow invocation of
static methods through type aliases, then this type definition would
essentially be switching the default hasher from `SipHasher` in libcollections
to a libstd-defined `RandomizedSipHasher` type. This type's `Default`
implementation would randomly seed the `SipHasher` instance, and otherwise
perform the same as `SipHasher`.
This future state doesn't seem incredibly far off, but until that time comes,
the hashmap module will live in libstd to not compromise on functionality.
* In preparation for the hashmap moving to libcollections, the `hash` module has
moved from libstd to libcollections. A previously snapshotted commit enables a
distinct `Writer` trait to live in the `hash` module which `Hash`
implementations are now parameterized over.
Due to using a custom trait, the `SipHasher` implementation has lost its
specialized methods for writing integers. These can be re-added
backwards-compatibly in the future via default methods if necessary, but the
FNV hashing should satisfy much of the need for speedier hashing.
A list of breaking changes:
* HashMap::{get, get_mut} no longer fails with the key formatted into the error
message with `{:?}`, instead, a generic message is printed. With backtraces,
it should still be not-too-hard to track down errors.
* The HashMap, HashSet, and LruCache types are now available through
std::collections instead of the collections crate.
* Manual implementations of hash should be parameterized over `hash::Writer`
instead of just `Writer`.
[breaking-change]
2014-05-29 18:50:12 -07:00
|
|
|
|
use clone::Clone;
|
|
|
|
|
use cmp::{max, Eq, Equiv, PartialEq};
|
|
|
|
|
use default::Default;
|
2014-10-23 08:42:21 -07:00
|
|
|
|
use fmt::{mod, Show};
|
2014-07-15 21:58:35 +01:00
|
|
|
|
use hash::{Hash, Hasher, RandomSipHasher};
|
2014-12-12 01:02:19 -06:00
|
|
|
|
use iter::{mod, Iterator, IteratorExt, FromIterator, Extend, Map};
|
2014-10-24 12:25:50 -05:00
|
|
|
|
use kinds::Sized;
|
2014-10-23 08:42:21 -07:00
|
|
|
|
use mem::{mod, replace};
|
2014-11-17 14:23:21 +01:00
|
|
|
|
use num::{Int, UnsignedInt};
|
2014-12-07 14:15:25 -05:00
|
|
|
|
use ops::{Deref, FnMut, Index, IndexMut};
|
2014-11-28 11:57:41 -05:00
|
|
|
|
use option::Option;
|
|
|
|
|
use option::Option::{Some, None};
|
|
|
|
|
use result::Result;
|
|
|
|
|
use result::Result::{Ok, Err};
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
use super::table::{
|
2014-12-13 11:15:18 -05:00
|
|
|
|
mod,
|
2014-07-15 21:58:35 +01:00
|
|
|
|
Bucket,
|
2014-09-18 17:05:22 -04:00
|
|
|
|
EmptyBucket,
|
2014-07-15 21:58:35 +01:00
|
|
|
|
FullBucket,
|
|
|
|
|
FullBucketImm,
|
|
|
|
|
FullBucketMut,
|
|
|
|
|
RawTable,
|
|
|
|
|
SafeHash
|
|
|
|
|
};
|
2014-12-13 11:15:18 -05:00
|
|
|
|
use super::table::BucketState::{
|
|
|
|
|
Empty,
|
|
|
|
|
Full,
|
|
|
|
|
};
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-10-06 16:29:47 -07:00
|
|
|
|
const INITIAL_LOG2_CAP: uint = 5;
|
|
|
|
|
pub const INITIAL_CAPACITY: uint = 1 << INITIAL_LOG2_CAP; // 2^5
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// The default behavior of HashMap implements a load factor of 90.9%.
|
2014-11-17 14:23:21 +01:00
|
|
|
|
/// This behavior is characterized by the following condition:
|
2014-07-16 00:39:32 +01:00
|
|
|
|
///
|
2014-11-17 14:23:21 +01:00
|
|
|
|
/// - if size > 0.909 * capacity: grow the map
|
2014-07-16 00:39:32 +01:00
|
|
|
|
#[deriving(Clone)]
|
2014-11-17 14:23:21 +01:00
|
|
|
|
struct DefaultResizePolicy;
|
2013-04-03 08:45:14 -04:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl DefaultResizePolicy {
|
2014-11-17 14:23:21 +01:00
|
|
|
|
fn new() -> DefaultResizePolicy {
|
|
|
|
|
DefaultResizePolicy
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2012-09-26 17:47:29 -07:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
#[inline]
|
2014-11-17 14:23:21 +01:00
|
|
|
|
fn min_capacity(&self, usable_size: uint) -> uint {
|
|
|
|
|
// Here, we are rephrasing the logic by specifying the lower limit
|
|
|
|
|
// on capacity:
|
2014-07-15 21:58:35 +01:00
|
|
|
|
//
|
2014-11-17 14:23:21 +01:00
|
|
|
|
// - if `cap < size * 1.1`: grow the map
|
|
|
|
|
usable_size * 11 / 10
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
/// An inverse of `min_capacity`, approximately.
|
2014-07-16 00:39:32 +01:00
|
|
|
|
#[inline]
|
2014-11-17 14:23:21 +01:00
|
|
|
|
fn usable_capacity(&self, cap: uint) -> uint {
|
|
|
|
|
// As the number of entries approaches usable capacity,
|
|
|
|
|
// min_capacity(size) must be smaller than the internal capacity,
|
|
|
|
|
// so that the map is not resized:
|
|
|
|
|
// `min_capacity(usable_capacity(x)) <= x`.
|
|
|
|
|
// The lef-hand side can only be smaller due to flooring by integer
|
|
|
|
|
// division.
|
|
|
|
|
//
|
|
|
|
|
// This doesn't have to be checked for overflow since allocation size
|
|
|
|
|
// in bytes will overflow earlier than multiplication by 10.
|
|
|
|
|
cap * 10 / 11
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_resize_policy() {
|
|
|
|
|
use prelude::*;
|
|
|
|
|
let rp = DefaultResizePolicy;
|
|
|
|
|
for n in range(0u, 1000) {
|
|
|
|
|
assert!(rp.min_capacity(rp.usable_capacity(n)) <= n);
|
|
|
|
|
assert!(rp.usable_capacity(rp.min_capacity(n)) <= n);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// The main performance trick in this hashmap is called Robin Hood Hashing.
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// It gains its excellent performance from one essential operation:
|
2014-07-16 00:39:32 +01:00
|
|
|
|
//
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// If an insertion collides with an existing element, and that element's
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// "probe distance" (how far away the element is from its ideal location)
|
|
|
|
|
// is higher than how far we've already probed, swap the elements.
|
|
|
|
|
//
|
|
|
|
|
// This massively lowers variance in probe distance, and allows us to get very
|
|
|
|
|
// high load factors with good performance. The 90% load factor I use is rather
|
|
|
|
|
// conservative.
|
|
|
|
|
//
|
|
|
|
|
// > Why a load factor of approximately 90%?
|
|
|
|
|
//
|
|
|
|
|
// In general, all the distances to initial buckets will converge on the mean.
|
|
|
|
|
// At a load factor of α, the odds of finding the target bucket after k
|
|
|
|
|
// probes is approximately 1-α^k. If we set this equal to 50% (since we converge
|
|
|
|
|
// on the mean) and set k=8 (64-byte cache line / 8-byte hash), α=0.92. I round
|
|
|
|
|
// this down to make the math easier on the CPU and avoid its FPU.
|
|
|
|
|
// Since on average we start the probing in the middle of a cache line, this
|
|
|
|
|
// strategy pulls in two cache lines of hashes on every lookup. I think that's
|
|
|
|
|
// pretty good, but if you want to trade off some space, it could go down to one
|
|
|
|
|
// cache line on average with an α of 0.84.
|
|
|
|
|
//
|
|
|
|
|
// > Wait, what? Where did you get 1-α^k from?
|
|
|
|
|
//
|
|
|
|
|
// On the first probe, your odds of a collision with an existing element is α.
|
|
|
|
|
// The odds of doing this twice in a row is approximately α^2. For three times,
|
|
|
|
|
// α^3, etc. Therefore, the odds of colliding k times is α^k. The odds of NOT
|
|
|
|
|
// colliding after k tries is 1-α^k.
|
|
|
|
|
//
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// The paper from 1986 cited below mentions an implementation which keeps track
|
|
|
|
|
// of the distance-to-initial-bucket histogram. This approach is not suitable
|
|
|
|
|
// for modern architectures because it requires maintaining an internal data
|
|
|
|
|
// structure. This allows very good first guesses, but we are most concerned
|
|
|
|
|
// with guessing entire cache lines, not individual indexes. Furthermore, array
|
|
|
|
|
// accesses are no longer linear and in one direction, as we have now. There
|
|
|
|
|
// is also memory and cache pressure that this would entail that would be very
|
|
|
|
|
// difficult to properly see in a microbenchmark.
|
|
|
|
|
//
|
2014-10-09 15:17:22 -04:00
|
|
|
|
// ## Future Improvements (FIXME!)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
//
|
|
|
|
|
// Allow the load factor to be changed dynamically and/or at initialization.
|
|
|
|
|
//
|
|
|
|
|
// Also, would it be possible for us to reuse storage when growing the
|
|
|
|
|
// underlying table? This is exactly the use case for 'realloc', and may
|
|
|
|
|
// be worth exploring.
|
|
|
|
|
//
|
2014-10-09 15:17:22 -04:00
|
|
|
|
// ## Future Optimizations (FIXME!)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
//
|
|
|
|
|
// Another possible design choice that I made without any real reason is
|
|
|
|
|
// parameterizing the raw table over keys and values. Technically, all we need
|
|
|
|
|
// is the size and alignment of keys and values, and the code should be just as
|
|
|
|
|
// efficient (well, we might need one for power-of-two size and one for not...).
|
|
|
|
|
// This has the potential to reduce code bloat in rust executables, without
|
|
|
|
|
// really losing anything except 4 words (key size, key alignment, val size,
|
|
|
|
|
// val alignment) which can be passed in to every call of a `RawTable` function.
|
|
|
|
|
// This would definitely be an avenue worth exploring if people start complaining
|
|
|
|
|
// about the size of rust executables.
|
|
|
|
|
//
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// Annotate exceedingly likely branches in `table::make_hash`
|
2014-11-12 14:55:51 -08:00
|
|
|
|
// and `search_hashed` to reduce instruction cache pressure
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// and mispredictions once it becomes possible (blocked on issue #11092).
|
|
|
|
|
//
|
|
|
|
|
// Shrinking the table could simply reallocate in place after moving buckets
|
|
|
|
|
// to the first half.
|
|
|
|
|
//
|
|
|
|
|
// The growth algorithm (fragment of the Proof of Correctness)
|
|
|
|
|
// --------------------
|
|
|
|
|
//
|
|
|
|
|
// The growth algorithm is basically a fast path of the naive reinsertion-
|
|
|
|
|
// during-resize algorithm. Other paths should never be taken.
|
|
|
|
|
//
|
|
|
|
|
// Consider growing a robin hood hashtable of capacity n. Normally, we do this
|
|
|
|
|
// by allocating a new table of capacity `2n`, and then individually reinsert
|
|
|
|
|
// each element in the old table into the new one. This guarantees that the
|
|
|
|
|
// new table is a valid robin hood hashtable with all the desired statistical
|
|
|
|
|
// properties. Remark that the order we reinsert the elements in should not
|
|
|
|
|
// matter. For simplicity and efficiency, we will consider only linear
|
|
|
|
|
// reinsertions, which consist of reinserting all elements in the old table
|
|
|
|
|
// into the new one by increasing order of index. However we will not be
|
|
|
|
|
// starting our reinsertions from index 0 in general. If we start from index
|
|
|
|
|
// i, for the purpose of reinsertion we will consider all elements with real
|
|
|
|
|
// index j < i to have virtual index n + j.
|
|
|
|
|
//
|
|
|
|
|
// Our hash generation scheme consists of generating a 64-bit hash and
|
|
|
|
|
// truncating the most significant bits. When moving to the new table, we
|
|
|
|
|
// simply introduce a new bit to the front of the hash. Therefore, if an
|
|
|
|
|
// elements has ideal index i in the old table, it can have one of two ideal
|
|
|
|
|
// locations in the new table. If the new bit is 0, then the new ideal index
|
|
|
|
|
// is i. If the new bit is 1, then the new ideal index is n + i. Intutively,
|
|
|
|
|
// we are producing two independent tables of size n, and for each element we
|
|
|
|
|
// independently choose which table to insert it into with equal probability.
|
|
|
|
|
// However the rather than wrapping around themselves on overflowing their
|
|
|
|
|
// indexes, the first table overflows into the first, and the first into the
|
|
|
|
|
// second. Visually, our new table will look something like:
|
|
|
|
|
//
|
|
|
|
|
// [yy_xxx_xxxx_xxx|xx_yyy_yyyy_yyy]
|
|
|
|
|
//
|
|
|
|
|
// Where x's are elements inserted into the first table, y's are elements
|
|
|
|
|
// inserted into the second, and _'s are empty sections. We now define a few
|
|
|
|
|
// key concepts that we will use later. Note that this is a very abstract
|
|
|
|
|
// perspective of the table. A real resized table would be at least half
|
|
|
|
|
// empty.
|
|
|
|
|
//
|
|
|
|
|
// Theorem: A linear robin hood reinsertion from the first ideal element
|
|
|
|
|
// produces identical results to a linear naive reinsertion from the same
|
|
|
|
|
// element.
|
|
|
|
|
//
|
|
|
|
|
// FIXME(Gankro, pczarn): review the proof and put it all in a separate doc.rs
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// A hash map implementation which uses linear probing with Robin
|
|
|
|
|
/// Hood bucket stealing.
|
|
|
|
|
///
|
|
|
|
|
/// The hashes are all keyed by the task-local random number generator
|
|
|
|
|
/// on creation by default. This means that the ordering of the keys is
|
|
|
|
|
/// randomized, but makes the tables more resistant to
|
|
|
|
|
/// denial-of-service attacks (Hash DoS). This behaviour can be
|
|
|
|
|
/// overridden with one of the constructors.
|
|
|
|
|
///
|
|
|
|
|
/// It is required that the keys implement the `Eq` and `Hash` traits, although
|
|
|
|
|
/// this can frequently be achieved by using `#[deriving(Eq, Hash)]`.
|
|
|
|
|
///
|
|
|
|
|
/// Relevant papers/articles:
|
|
|
|
|
///
|
|
|
|
|
/// 1. Pedro Celis. ["Robin Hood Hashing"](https://cs.uwaterloo.ca/research/tr/1986/CS-86-14.pdf)
|
|
|
|
|
/// 2. Emmanuel Goossaert. ["Robin Hood
|
|
|
|
|
/// hashing"](http://codecapsule.com/2013/11/11/robin-hood-hashing/)
|
|
|
|
|
/// 3. Emmanuel Goossaert. ["Robin Hood hashing: backward shift
|
|
|
|
|
/// deletion"](http://codecapsule.com/2013/11/17/robin-hood-hashing-backward-shift-deletion/)
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// // type inference lets us omit an explicit type signature (which
|
|
|
|
|
/// // would be `HashMap<&str, &str>` in this example).
|
|
|
|
|
/// let mut book_reviews = HashMap::new();
|
|
|
|
|
///
|
|
|
|
|
/// // review some books.
|
|
|
|
|
/// book_reviews.insert("Adventures of Huckleberry Finn", "My favorite book.");
|
|
|
|
|
/// book_reviews.insert("Grimms' Fairy Tales", "Masterpiece.");
|
|
|
|
|
/// book_reviews.insert("Pride and Prejudice", "Very enjoyable.");
|
|
|
|
|
/// book_reviews.insert("The Adventures of Sherlock Holmes", "Eye lyked it alot.");
|
|
|
|
|
///
|
|
|
|
|
/// // check for a specific one.
|
|
|
|
|
/// if !book_reviews.contains_key(&("Les Misérables")) {
|
|
|
|
|
/// println!("We've got {} reviews, but Les Misérables ain't one.",
|
|
|
|
|
/// book_reviews.len());
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// // oops, this review has a lot of spelling mistakes, let's delete it.
|
|
|
|
|
/// book_reviews.remove(&("The Adventures of Sherlock Holmes"));
|
|
|
|
|
///
|
|
|
|
|
/// // look up the values associated with some keys.
|
|
|
|
|
/// let to_find = ["Pride and Prejudice", "Alice's Adventure in Wonderland"];
|
|
|
|
|
/// for book in to_find.iter() {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// match book_reviews.get(book) {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Some(review) => println!("{}: {}", *book, *review),
|
|
|
|
|
/// None => println!("{} is unreviewed.", *book)
|
|
|
|
|
/// }
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// // iterate over everything.
|
|
|
|
|
/// for (book, review) in book_reviews.iter() {
|
|
|
|
|
/// println!("{}: \"{}\"", *book, *review);
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
///
|
|
|
|
|
/// The easiest way to use `HashMap` with a custom type is to derive `Eq` and `Hash`.
|
|
|
|
|
/// We must also derive `PartialEq`.
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// #[deriving(Hash, Eq, PartialEq, Show)]
|
|
|
|
|
/// struct Viking<'a> {
|
|
|
|
|
/// name: &'a str,
|
|
|
|
|
/// power: uint,
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// let mut vikings = HashMap::new();
|
|
|
|
|
///
|
|
|
|
|
/// vikings.insert("Norway", Viking { name: "Einar", power: 9u });
|
|
|
|
|
/// vikings.insert("Denmark", Viking { name: "Olaf", power: 4u });
|
|
|
|
|
/// vikings.insert("Iceland", Viking { name: "Harald", power: 8u });
|
|
|
|
|
///
|
|
|
|
|
/// // Use derived implementation to print the vikings.
|
|
|
|
|
/// for (land, viking) in vikings.iter() {
|
|
|
|
|
/// println!("{} at {}", viking, land);
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
|
|
|
|
#[deriving(Clone)]
|
|
|
|
|
pub struct HashMap<K, V, H = RandomSipHasher> {
|
|
|
|
|
// All hashes are keyed on these values, to prevent hash collision attacks.
|
|
|
|
|
hasher: H,
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
table: RawTable<K, V>,
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
resize_policy: DefaultResizePolicy,
|
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Search for a pre-hashed key.
|
2014-12-07 14:15:25 -05:00
|
|
|
|
fn search_hashed<K, V, M, F>(table: M,
|
2014-12-16 16:00:37 -05:00
|
|
|
|
hash: SafeHash,
|
2014-12-07 14:15:25 -05:00
|
|
|
|
mut is_match: F)
|
|
|
|
|
-> SearchResult<K, V, M> where
|
|
|
|
|
M: Deref<RawTable<K, V>>,
|
|
|
|
|
F: FnMut(&K) -> bool,
|
|
|
|
|
{
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let size = table.size();
|
|
|
|
|
let mut probe = Bucket::new(table, hash);
|
|
|
|
|
let ib = probe.index();
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
while probe.index() != ib + size {
|
|
|
|
|
let full = match probe.peek() {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
Empty(b) => return TableRef(b.into_table()), // hit an empty bucket
|
|
|
|
|
Full(b) => b
|
2014-07-16 00:39:32 +01:00
|
|
|
|
};
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
if full.distance() + ib < full.index() {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// We can finish the search early if we hit any bucket
|
|
|
|
|
// with a lower distance to initial bucket than we've probed.
|
|
|
|
|
return TableRef(full.into_table());
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// If the hash doesn't match, it can't be this one..
|
2014-12-16 16:00:37 -05:00
|
|
|
|
if hash == full.hash() {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// If the key doesn't match, it can't be this one..
|
2014-12-16 16:00:37 -05:00
|
|
|
|
if is_match(full.read().0) {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
return FoundExisting(full);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
probe = full.next();
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
TableRef(probe.into_table())
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-09-18 17:05:22 -04:00
|
|
|
|
fn pop_internal<K, V>(starting_bucket: FullBucketMut<K, V>) -> (K, V) {
|
|
|
|
|
let (empty, retkey, retval) = starting_bucket.take();
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let mut gap = match empty.gap_peek() {
|
|
|
|
|
Some(b) => b,
|
2014-09-18 17:05:22 -04:00
|
|
|
|
None => return (retkey, retval)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
};
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
while gap.full().distance() != 0 {
|
|
|
|
|
gap = match gap.shift() {
|
|
|
|
|
Some(b) => b,
|
|
|
|
|
None => break
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Now we've done all our shifting. Return the value we grabbed earlier.
|
2014-12-16 16:00:37 -05:00
|
|
|
|
(retkey, retval)
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Perform robin hood bucket stealing at the given `bucket`. You must
|
|
|
|
|
/// also pass the position of that bucket's initial bucket so we don't have
|
|
|
|
|
/// to recalculate it.
|
|
|
|
|
///
|
|
|
|
|
/// `hash`, `k`, and `v` are the elements to "robin hood" into the hashtable.
|
|
|
|
|
fn robin_hood<'a, K: 'a, V: 'a>(mut bucket: FullBucketMut<'a, K, V>,
|
|
|
|
|
mut ib: uint,
|
|
|
|
|
mut hash: SafeHash,
|
|
|
|
|
mut k: K,
|
|
|
|
|
mut v: V)
|
|
|
|
|
-> &'a mut V {
|
|
|
|
|
let starting_index = bucket.index();
|
|
|
|
|
let size = {
|
|
|
|
|
let table = bucket.table(); // FIXME "lifetime too short".
|
|
|
|
|
table.size()
|
|
|
|
|
};
|
|
|
|
|
// There can be at most `size - dib` buckets to displace, because
|
|
|
|
|
// in the worst case, there are `size` elements and we already are
|
|
|
|
|
// `distance` buckets away from the initial one.
|
|
|
|
|
let idx_end = starting_index + size - bucket.distance();
|
|
|
|
|
|
|
|
|
|
loop {
|
|
|
|
|
let (old_hash, old_key, old_val) = bucket.replace(hash, k, v);
|
|
|
|
|
loop {
|
|
|
|
|
let probe = bucket.next();
|
|
|
|
|
assert!(probe.index() != idx_end);
|
|
|
|
|
|
|
|
|
|
let full_bucket = match probe.peek() {
|
2014-12-13 11:15:18 -05:00
|
|
|
|
Empty(bucket) => {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// Found a hole!
|
|
|
|
|
let b = bucket.put(old_hash, old_key, old_val);
|
|
|
|
|
// Now that it's stolen, just read the value's pointer
|
|
|
|
|
// right out of the table!
|
2014-12-16 16:00:37 -05:00
|
|
|
|
return Bucket::at_index(b.into_table(), starting_index)
|
|
|
|
|
.peek()
|
|
|
|
|
.expect_full()
|
|
|
|
|
.into_mut_refs()
|
|
|
|
|
.1;
|
2014-07-15 21:58:35 +01:00
|
|
|
|
},
|
2014-12-13 11:15:18 -05:00
|
|
|
|
Full(bucket) => bucket
|
2014-07-16 00:39:32 +01:00
|
|
|
|
};
|
2014-07-15 21:58:35 +01:00
|
|
|
|
|
|
|
|
|
let probe_ib = full_bucket.index() - full_bucket.distance();
|
|
|
|
|
|
|
|
|
|
bucket = full_bucket;
|
|
|
|
|
|
|
|
|
|
// Robin hood! Steal the spot.
|
|
|
|
|
if ib < probe_ib {
|
|
|
|
|
ib = probe_ib;
|
|
|
|
|
hash = old_hash;
|
|
|
|
|
k = old_key;
|
|
|
|
|
v = old_val;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A result that works like Option<FullBucket<..>> but preserves
|
|
|
|
|
/// the reference that grants us access to the table in any case.
|
|
|
|
|
enum SearchResult<K, V, M> {
|
|
|
|
|
// This is an entry that holds the given key:
|
|
|
|
|
FoundExisting(FullBucket<K, V, M>),
|
|
|
|
|
|
|
|
|
|
// There was no such entry. The reference is given back:
|
|
|
|
|
TableRef(M)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<K, V, M> SearchResult<K, V, M> {
|
|
|
|
|
fn into_option(self) -> Option<FullBucket<K, V, M>> {
|
|
|
|
|
match self {
|
|
|
|
|
FoundExisting(bucket) => Some(bucket),
|
|
|
|
|
TableRef(_) => None
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V, S, H: Hasher<S>> HashMap<K, V, H> {
|
2014-10-24 12:25:50 -05:00
|
|
|
|
fn make_hash<Sized? X: Hash<S>>(&self, x: &X) -> SafeHash {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
table::make_hash(&self.hasher, x)
|
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-11-26 23:50:12 -05:00
|
|
|
|
#[allow(deprecated)]
|
2014-10-24 12:25:50 -05:00
|
|
|
|
fn search_equiv<'a, Sized? Q: Hash<S> + Equiv<K>>(&'a self, q: &Q)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
-> Option<FullBucketImm<'a, K, V>> {
|
|
|
|
|
let hash = self.make_hash(q);
|
2014-12-16 16:00:37 -05:00
|
|
|
|
search_hashed(&self.table, hash, |k| q.equiv(k)).into_option()
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-11-26 23:50:12 -05:00
|
|
|
|
#[allow(deprecated)]
|
2014-10-24 12:25:50 -05:00
|
|
|
|
fn search_equiv_mut<'a, Sized? Q: Hash<S> + Equiv<K>>(&'a mut self, q: &Q)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
-> Option<FullBucketMut<'a, K, V>> {
|
|
|
|
|
let hash = self.make_hash(q);
|
2014-12-16 16:00:37 -05:00
|
|
|
|
search_hashed(&mut self.table, hash, |k| q.equiv(k)).into_option()
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Search for a key, yielding the index if it's found in the hashtable.
|
|
|
|
|
/// If you already have the hash for the key lying around, use
|
|
|
|
|
/// search_hashed.
|
2014-11-12 14:55:51 -08:00
|
|
|
|
fn search<'a, Sized? Q>(&'a self, q: &Q) -> Option<FullBucketImm<'a, K, V>>
|
|
|
|
|
where Q: BorrowFrom<K> + Eq + Hash<S>
|
|
|
|
|
{
|
|
|
|
|
let hash = self.make_hash(q);
|
2014-12-16 16:00:37 -05:00
|
|
|
|
search_hashed(&self.table, hash, |k| q.eq(BorrowFrom::borrow_from(k)))
|
2014-11-12 14:55:51 -08:00
|
|
|
|
.into_option()
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
fn search_mut<'a, Sized? Q>(&'a mut self, q: &Q) -> Option<FullBucketMut<'a, K, V>>
|
|
|
|
|
where Q: BorrowFrom<K> + Eq + Hash<S>
|
|
|
|
|
{
|
|
|
|
|
let hash = self.make_hash(q);
|
2014-12-16 16:00:37 -05:00
|
|
|
|
search_hashed(&mut self.table, hash, |k| q.eq(BorrowFrom::borrow_from(k)))
|
2014-11-12 14:55:51 -08:00
|
|
|
|
.into_option()
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// The caller should ensure that invariants by Robin Hood Hashing hold.
|
|
|
|
|
fn insert_hashed_ordered(&mut self, hash: SafeHash, k: K, v: V) {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let cap = self.table.capacity();
|
2014-12-16 16:00:37 -05:00
|
|
|
|
let mut buckets = Bucket::new(&mut self.table, hash);
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let ib = buckets.index();
|
2014-07-15 21:58:35 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
while buckets.index() != ib + cap {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// We don't need to compare hashes for value swap.
|
|
|
|
|
// Not even DIBs for Robin Hood.
|
2014-07-16 00:39:32 +01:00
|
|
|
|
buckets = match buckets.peek() {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
Empty(empty) => {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
empty.put(hash, k, v);
|
|
|
|
|
return;
|
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
Full(b) => b.into_bucket()
|
2014-07-16 00:39:32 +01:00
|
|
|
|
};
|
|
|
|
|
buckets.next();
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-10-09 15:17:22 -04:00
|
|
|
|
panic!("Internal HashMap error: Out of space.");
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Hash + Eq, V> HashMap<K, V, RandomSipHasher> {
|
|
|
|
|
/// Create an empty HashMap.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
2014-11-05 22:11:52 +02:00
|
|
|
|
/// let mut map: HashMap<&str, int> = HashMap::new();
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
pub fn new() -> HashMap<K, V, RandomSipHasher> {
|
|
|
|
|
let hasher = RandomSipHasher::new();
|
|
|
|
|
HashMap::with_hasher(hasher)
|
2012-07-26 16:09:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Creates an empty hash map with the given initial capacity.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
/// let mut map: HashMap<&str, int> = HashMap::with_capacity(10);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
pub fn with_capacity(capacity: uint) -> HashMap<K, V, RandomSipHasher> {
|
|
|
|
|
let hasher = RandomSipHasher::new();
|
|
|
|
|
HashMap::with_capacity_and_hasher(capacity, hasher)
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V, S, H: Hasher<S>> HashMap<K, V, H> {
|
|
|
|
|
/// Creates an empty hashmap which will use the given hasher to hash keys.
|
|
|
|
|
///
|
|
|
|
|
/// The creates map has the default initial capacity.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
/// use std::hash::sip::SipHasher;
|
|
|
|
|
///
|
|
|
|
|
/// let h = SipHasher::new();
|
|
|
|
|
/// let mut map = HashMap::with_hasher(h);
|
|
|
|
|
/// map.insert(1i, 2u);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn with_hasher(hasher: H) -> HashMap<K, V, H> {
|
|
|
|
|
HashMap {
|
|
|
|
|
hasher: hasher,
|
2014-11-17 14:23:21 +01:00
|
|
|
|
resize_policy: DefaultResizePolicy::new(),
|
2014-07-15 21:58:35 +01:00
|
|
|
|
table: RawTable::new(0),
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Create an empty HashMap with space for at least `capacity`
|
|
|
|
|
/// elements, using `hasher` to hash the keys.
|
|
|
|
|
///
|
|
|
|
|
/// Warning: `hasher` is normally randomly generated, and
|
|
|
|
|
/// is designed to allow HashMaps to be resistant to attacks that
|
|
|
|
|
/// cause many collisions and very poor performance. Setting it
|
|
|
|
|
/// manually using this function can expose a DoS attack vector.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
/// use std::hash::sip::SipHasher;
|
|
|
|
|
///
|
|
|
|
|
/// let h = SipHasher::new();
|
|
|
|
|
/// let mut map = HashMap::with_capacity_and_hasher(10, h);
|
|
|
|
|
/// map.insert(1i, 2u);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
|
|
|
|
pub fn with_capacity_and_hasher(capacity: uint, hasher: H) -> HashMap<K, V, H> {
|
2014-11-17 14:23:21 +01:00
|
|
|
|
let resize_policy = DefaultResizePolicy::new();
|
|
|
|
|
let min_cap = max(INITIAL_CAPACITY, resize_policy.min_capacity(capacity));
|
|
|
|
|
let internal_cap = min_cap.checked_next_power_of_two().expect("capacity overflow");
|
|
|
|
|
assert!(internal_cap >= capacity, "capacity overflow");
|
2014-07-16 00:39:32 +01:00
|
|
|
|
HashMap {
|
|
|
|
|
hasher: hasher,
|
2014-11-17 14:23:21 +01:00
|
|
|
|
resize_policy: resize_policy,
|
|
|
|
|
table: RawTable::new(internal_cap),
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-03-26 21:58:08 -04:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
/// Returns the number of elements the map can hold without reallocating.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
/// let map: HashMap<int, int> = HashMap::with_capacity(100);
|
|
|
|
|
/// assert!(map.capacity() >= 100);
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
|
|
|
|
pub fn capacity(&self) -> uint {
|
|
|
|
|
self.resize_policy.usable_capacity(self.table.capacity())
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-08 17:26:52 +01:00
|
|
|
|
/// Reserves capacity for at least `additional` more elements to be inserted
|
|
|
|
|
/// in the `HashMap`. The collection may reserve more space to avoid
|
|
|
|
|
/// frequent reallocations.
|
2014-07-16 00:39:32 +01:00
|
|
|
|
///
|
2014-11-08 17:26:52 +01:00
|
|
|
|
/// # Panics
|
|
|
|
|
///
|
|
|
|
|
/// Panics if the new allocation size overflows `uint`.
|
2014-07-16 00:39:32 +01:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
/// let mut map: HashMap<&str, int> = HashMap::new();
|
|
|
|
|
/// map.reserve(10);
|
|
|
|
|
/// ```
|
2014-11-08 17:26:52 +01:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
|
|
|
|
pub fn reserve(&mut self, additional: uint) {
|
|
|
|
|
let new_size = self.len().checked_add(additional).expect("capacity overflow");
|
|
|
|
|
let min_cap = self.resize_policy.min_capacity(new_size);
|
|
|
|
|
|
|
|
|
|
// An invalid value shouldn't make us run out of space. This includes
|
|
|
|
|
// an overflow check.
|
|
|
|
|
assert!(new_size <= min_cap);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-11-08 17:26:52 +01:00
|
|
|
|
if self.table.capacity() < min_cap {
|
|
|
|
|
let new_capacity = max(min_cap.next_power_of_two(), INITIAL_CAPACITY);
|
|
|
|
|
self.resize(new_capacity);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Resizes the internal vectors to a new capacity. It's your responsibility to:
|
|
|
|
|
/// 1) Make sure the new capacity is enough for all the elements, accounting
|
|
|
|
|
/// for the load factor.
|
2014-12-08 01:03:35 -05:00
|
|
|
|
/// 2) Ensure new_capacity is a power of two or zero.
|
2014-07-16 00:39:32 +01:00
|
|
|
|
fn resize(&mut self, new_capacity: uint) {
|
|
|
|
|
assert!(self.table.size() <= new_capacity);
|
2014-12-08 01:03:35 -05:00
|
|
|
|
assert!(new_capacity.is_power_of_two() || new_capacity == 0);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
let mut old_table = replace(&mut self.table, RawTable::new(new_capacity));
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let old_size = old_table.size();
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
if old_table.capacity() == 0 || old_table.size() == 0 {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
return;
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
// Grow the table.
|
|
|
|
|
// Specialization of the other branch.
|
|
|
|
|
let mut bucket = Bucket::first(&mut old_table);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
// "So a few of the first shall be last: for many be called,
|
|
|
|
|
// but few chosen."
|
|
|
|
|
//
|
|
|
|
|
// We'll most likely encounter a few buckets at the beginning that
|
|
|
|
|
// have their initial buckets near the end of the table. They were
|
|
|
|
|
// placed at the beginning as the probe wrapped around the table
|
|
|
|
|
// during insertion. We must skip forward to a bucket that won't
|
|
|
|
|
// get reinserted too early and won't unfairly steal others spot.
|
|
|
|
|
// This eliminates the need for robin hood.
|
|
|
|
|
loop {
|
|
|
|
|
bucket = match bucket.peek() {
|
|
|
|
|
Full(full) => {
|
|
|
|
|
if full.distance() == 0 {
|
|
|
|
|
// This bucket occupies its ideal spot.
|
|
|
|
|
// It indicates the start of another "cluster".
|
|
|
|
|
bucket = full.into_bucket();
|
|
|
|
|
break;
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-11-17 14:23:21 +01:00
|
|
|
|
// Leaving this bucket in the last cluster for later.
|
|
|
|
|
full.into_bucket()
|
|
|
|
|
}
|
|
|
|
|
Empty(b) => {
|
|
|
|
|
// Encountered a hole between clusters.
|
|
|
|
|
b.into_bucket()
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
bucket.next();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This is how the buckets might be laid out in memory:
|
|
|
|
|
// ($ marks an initialized bucket)
|
|
|
|
|
// ________________
|
|
|
|
|
// |$$$_$$$$$$_$$$$$|
|
|
|
|
|
//
|
|
|
|
|
// But we've skipped the entire initial cluster of buckets
|
|
|
|
|
// and will continue iteration in this order:
|
|
|
|
|
// ________________
|
|
|
|
|
// |$$$$$$_$$$$$
|
|
|
|
|
// ^ wrap around once end is reached
|
|
|
|
|
// ________________
|
|
|
|
|
// $$$_____________|
|
|
|
|
|
// ^ exit once table.size == 0
|
|
|
|
|
loop {
|
|
|
|
|
bucket = match bucket.peek() {
|
|
|
|
|
Full(bucket) => {
|
|
|
|
|
let h = bucket.hash();
|
|
|
|
|
let (b, k, v) = bucket.take();
|
|
|
|
|
self.insert_hashed_ordered(h, k, v);
|
|
|
|
|
{
|
|
|
|
|
let t = b.table(); // FIXME "lifetime too short".
|
|
|
|
|
if t.size() == 0 { break }
|
|
|
|
|
};
|
|
|
|
|
b.into_bucket()
|
|
|
|
|
}
|
|
|
|
|
Empty(b) => b.into_bucket()
|
|
|
|
|
};
|
|
|
|
|
bucket.next();
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
assert_eq!(self.table.size(), old_size);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
/// Shrinks the capacity of the map as much as possible. It will drop
|
|
|
|
|
/// down as much as possible while maintaining the internal rules
|
|
|
|
|
/// and possibly leaving some space in accordance with the resize policy.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map: HashMap<int, int> = HashMap::with_capacity(100);
|
|
|
|
|
/// map.insert(1, 2);
|
|
|
|
|
/// map.insert(3, 4);
|
|
|
|
|
/// assert!(map.capacity() >= 100);
|
|
|
|
|
/// map.shrink_to_fit();
|
|
|
|
|
/// assert!(map.capacity() >= 2);
|
|
|
|
|
/// ```
|
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
|
|
|
|
pub fn shrink_to_fit(&mut self) {
|
|
|
|
|
let min_capacity = self.resize_policy.min_capacity(self.len());
|
|
|
|
|
let min_capacity = max(min_capacity.next_power_of_two(), INITIAL_CAPACITY);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// An invalid value shouldn't make us run out of space.
|
2014-11-17 14:23:21 +01:00
|
|
|
|
debug_assert!(self.len() <= min_capacity);
|
|
|
|
|
|
|
|
|
|
if self.table.capacity() != min_capacity {
|
|
|
|
|
let old_table = replace(&mut self.table, RawTable::new(min_capacity));
|
|
|
|
|
let old_size = old_table.size();
|
|
|
|
|
|
|
|
|
|
// Shrink the table. Naive algorithm for resizing:
|
|
|
|
|
for (h, k, v) in old_table.into_iter() {
|
|
|
|
|
self.insert_hashed_nocheck(h, k, v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
debug_assert_eq!(self.table.size(), old_size);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-04 19:43:14 -08:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Insert a pre-hashed key-value pair, without first checking
|
|
|
|
|
/// that there's enough room in the buckets. Returns a reference to the
|
|
|
|
|
/// newly insert value.
|
|
|
|
|
///
|
|
|
|
|
/// If the key already exists, the hashtable will be returned untouched
|
|
|
|
|
/// and a reference to the existing element will be returned.
|
2014-07-15 21:58:35 +01:00
|
|
|
|
fn insert_hashed_nocheck(&mut self, hash: SafeHash, k: K, v: V) -> &mut V {
|
|
|
|
|
self.insert_or_replace_with(hash, k, v, |_, _, _| ())
|
2014-08-27 21:46:52 -04:00
|
|
|
|
}
|
|
|
|
|
|
2014-12-07 14:15:25 -05:00
|
|
|
|
fn insert_or_replace_with<'a, F>(&'a mut self,
|
|
|
|
|
hash: SafeHash,
|
|
|
|
|
k: K,
|
|
|
|
|
v: V,
|
|
|
|
|
mut found_existing: F)
|
|
|
|
|
-> &'a mut V where
|
|
|
|
|
F: FnMut(&mut K, &mut V, V),
|
|
|
|
|
{
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// Worst case, we'll find one empty bucket among `size + 1` buckets.
|
|
|
|
|
let size = self.table.size();
|
2014-12-16 16:00:37 -05:00
|
|
|
|
let mut probe = Bucket::new(&mut self.table, hash);
|
2014-07-15 21:58:35 +01:00
|
|
|
|
let ib = probe.index();
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
loop {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
let mut bucket = match probe.peek() {
|
|
|
|
|
Empty(bucket) => {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// Found a hole!
|
2014-12-16 16:00:37 -05:00
|
|
|
|
return bucket.put(hash, k, v).into_mut_refs().1;
|
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
Full(bucket) => bucket
|
2014-07-16 00:39:32 +01:00
|
|
|
|
};
|
2012-07-26 16:09:22 -07:00
|
|
|
|
|
2014-12-16 16:00:37 -05:00
|
|
|
|
// hash matches?
|
2014-07-16 00:39:32 +01:00
|
|
|
|
if bucket.hash() == hash {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
// key matches?
|
|
|
|
|
if k == *bucket.read_mut().0 {
|
2014-07-15 21:58:35 +01:00
|
|
|
|
let (bucket_k, bucket_v) = bucket.into_mut_refs();
|
|
|
|
|
debug_assert!(k == *bucket_k);
|
2014-07-16 00:39:32 +01:00
|
|
|
|
// Key already exists. Get its reference.
|
2014-07-15 21:58:35 +01:00
|
|
|
|
found_existing(bucket_k, bucket_v, v);
|
|
|
|
|
return bucket_v;
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let robin_ib = bucket.index() as int - bucket.distance() as int;
|
2013-04-10 13:11:35 -07:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
if (ib as int) < robin_ib {
|
|
|
|
|
// Found a luckier bucket than me. Better steal his spot.
|
2014-07-15 21:58:35 +01:00
|
|
|
|
return robin_hood(bucket, robin_ib as uint, hash, k, v);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
|
|
|
|
|
probe = bucket.next();
|
|
|
|
|
assert!(probe.index() != ib + size + 1);
|
2014-06-02 10:26:02 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// Deprecated: use `contains_key` and `BorrowFrom` instead.
|
|
|
|
|
#[deprecated = "use contains_key and BorrowFrom instead"]
|
2014-10-24 12:25:50 -05:00
|
|
|
|
pub fn contains_key_equiv<Sized? Q: Hash<S> + Equiv<K>>(&self, key: &Q) -> bool {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
self.search_equiv(key).is_some()
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// Deprecated: use `get` and `BorrowFrom` instead.
|
|
|
|
|
#[deprecated = "use get and BorrowFrom instead"]
|
2014-10-24 12:25:50 -05:00
|
|
|
|
pub fn find_equiv<'a, Sized? Q: Hash<S> + Equiv<K>>(&'a self, k: &Q) -> Option<&'a V> {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.search_equiv(k).map(|bucket| bucket.into_refs().1)
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-21 15:41:37 -04:00
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// Deprecated: use `remove` and `BorrowFrom` instead.
|
|
|
|
|
#[deprecated = "use remove and BorrowFrom instead"]
|
2014-10-24 12:25:50 -05:00
|
|
|
|
pub fn pop_equiv<Sized? Q:Hash<S> + Equiv<K>>(&mut self, k: &Q) -> Option<V> {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
if self.table.size() == 0 {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-17 14:23:21 +01:00
|
|
|
|
self.reserve(1);
|
2014-07-16 00:39:32 +01:00
|
|
|
|
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.search_equiv_mut(k).map(|bucket| pop_internal(bucket).1)
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// An iterator visiting all keys in arbitrary order.
|
|
|
|
|
/// Iterator element type is `&'a K`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert("a", 1i);
|
|
|
|
|
/// map.insert("b", 2);
|
|
|
|
|
/// map.insert("c", 3);
|
|
|
|
|
///
|
|
|
|
|
/// for key in map.keys() {
|
|
|
|
|
/// println!("{}", key);
|
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub fn keys(&self) -> Keys<K, V> {
|
2014-12-02 14:28:35 -05:00
|
|
|
|
fn first<A, B>((a, _): (A, B)) -> A { a }
|
|
|
|
|
|
2014-12-12 01:02:19 -06:00
|
|
|
|
Keys { inner: self.iter().map(first) }
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// An iterator visiting all values in arbitrary order.
|
|
|
|
|
/// Iterator element type is `&'a V`.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
2014-07-17 20:40:39 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert("a", 1i);
|
|
|
|
|
/// map.insert("b", 2);
|
|
|
|
|
/// map.insert("c", 3);
|
2014-07-17 20:40:39 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// for key in map.values() {
|
|
|
|
|
/// println!("{}", key);
|
|
|
|
|
/// }
|
2014-07-17 20:40:39 +02:00
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub fn values(&self) -> Values<K, V> {
|
2014-12-02 14:28:35 -05:00
|
|
|
|
fn second<A, B>((_, b): (A, B)) -> B { b }
|
|
|
|
|
|
2014-12-12 01:02:19 -06:00
|
|
|
|
Values { inner: self.iter().map(second) }
|
2013-05-11 17:42:59 +02:00
|
|
|
|
}
|
2013-06-21 17:05:05 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// An iterator visiting all key-value pairs in arbitrary order.
|
|
|
|
|
/// Iterator element type is `(&'a K, &'a V)`.
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
2014-07-19 05:24:56 +02:00
|
|
|
|
/// ```
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert("a", 1i);
|
|
|
|
|
/// map.insert("b", 2);
|
|
|
|
|
/// map.insert("c", 3);
|
|
|
|
|
///
|
|
|
|
|
/// for (key, val) in map.iter() {
|
|
|
|
|
/// println!("key: {} val: {}", key, val);
|
2014-07-14 19:29:29 +02:00
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub fn iter(&self) -> Entries<K, V> {
|
|
|
|
|
Entries { inner: self.table.iter() }
|
2013-06-21 17:05:05 +02:00
|
|
|
|
}
|
2013-07-16 03:55:52 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// An iterator visiting all key-value pairs in arbitrary order,
|
|
|
|
|
/// with mutable references to the values.
|
|
|
|
|
/// Iterator element type is `(&'a K, &'a mut V)`.
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
2014-07-19 05:24:56 +02:00
|
|
|
|
/// ```
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// use std::collections::HashMap;
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert("a", 1i);
|
|
|
|
|
/// map.insert("b", 2);
|
|
|
|
|
/// map.insert("c", 3);
|
|
|
|
|
///
|
|
|
|
|
/// // Update all values
|
2014-09-14 20:27:36 -07:00
|
|
|
|
/// for (_, val) in map.iter_mut() {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// *val *= 2;
|
|
|
|
|
/// }
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// for (key, val) in map.iter() {
|
|
|
|
|
/// println!("key: {} val: {}", key, val);
|
2014-07-14 19:29:29 +02:00
|
|
|
|
/// }
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-09-14 15:57:55 -07:00
|
|
|
|
pub fn iter_mut(&mut self) -> MutEntries<K, V> {
|
2014-09-14 20:27:36 -07:00
|
|
|
|
MutEntries { inner: self.table.iter_mut() }
|
2013-07-16 16:39:24 +02:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// Creates a consuming iterator, that is, one that moves each key-value
|
|
|
|
|
/// pair out of the map in arbitrary order. The map cannot be used after
|
|
|
|
|
/// calling this.
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
2014-07-19 05:24:56 +02:00
|
|
|
|
/// ```
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// use std::collections::HashMap;
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert("a", 1i);
|
|
|
|
|
/// map.insert("b", 2);
|
|
|
|
|
/// map.insert("c", 3);
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// // Not possible with .iter()
|
2014-09-14 20:27:36 -07:00
|
|
|
|
/// let vec: Vec<(&str, int)> = map.into_iter().collect();
|
2014-07-14 19:29:29 +02:00
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-09-14 15:57:55 -07:00
|
|
|
|
pub fn into_iter(self) -> MoveEntries<K, V> {
|
2014-12-02 14:28:35 -05:00
|
|
|
|
fn last_two<A, B, C>((_, b, c): (A, B, C)) -> (B, C) { (b, c) }
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
MoveEntries {
|
2014-12-02 14:28:35 -05:00
|
|
|
|
inner: self.table.into_iter().map(last_two)
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
2013-07-16 03:55:52 +02:00
|
|
|
|
}
|
2014-09-18 17:05:22 -04:00
|
|
|
|
|
|
|
|
|
/// Gets the given key's corresponding entry in the map for in-place manipulation
|
|
|
|
|
pub fn entry<'a>(&'a mut self, key: K) -> Entry<'a, K, V> {
|
2014-11-17 14:23:21 +01:00
|
|
|
|
// Gotta resize now.
|
|
|
|
|
self.reserve(1);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
|
|
|
|
|
let hash = self.make_hash(&key);
|
|
|
|
|
search_entry_hashed(&mut self.table, hash, key)
|
|
|
|
|
}
|
2014-10-30 13:43:24 -07:00
|
|
|
|
|
|
|
|
|
/// Return the number of elements in the map.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut a = HashMap::new();
|
|
|
|
|
/// assert_eq!(a.len(), 0);
|
|
|
|
|
/// a.insert(1u, "a");
|
|
|
|
|
/// assert_eq!(a.len(), 1);
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-10-30 13:43:24 -07:00
|
|
|
|
pub fn len(&self) -> uint { self.table.size() }
|
|
|
|
|
|
|
|
|
|
/// Return true if the map contains no elements.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut a = HashMap::new();
|
|
|
|
|
/// assert!(a.is_empty());
|
|
|
|
|
/// a.insert(1u, "a");
|
|
|
|
|
/// assert!(!a.is_empty());
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-10-30 13:43:24 -07:00
|
|
|
|
pub fn is_empty(&self) -> bool { self.len() == 0 }
|
|
|
|
|
|
2014-12-16 17:45:03 -05:00
|
|
|
|
/// Clears the map, returning all key-value pairs as an iterator. Keeps the
|
|
|
|
|
/// allocated memory for reuse.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut a = HashMap::new();
|
|
|
|
|
/// a.insert(1u, "a");
|
|
|
|
|
/// a.insert(2u, "b");
|
|
|
|
|
///
|
|
|
|
|
/// for (k, v) in a.drain().take(1) {
|
|
|
|
|
/// assert!(k == 1 || k == 2);
|
|
|
|
|
/// assert!(v == "a" || v == "b");
|
|
|
|
|
/// }
|
|
|
|
|
///
|
|
|
|
|
/// assert!(a.is_empty());
|
|
|
|
|
/// ```
|
|
|
|
|
#[inline]
|
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
|
|
|
|
pub fn drain(&mut self) -> Drain<K, V> {
|
|
|
|
|
fn last_two<A, B, C>((_, b, c): (A, B, C)) -> (B, C) { (b, c) }
|
|
|
|
|
|
|
|
|
|
Drain {
|
|
|
|
|
inner: self.table.drain().map(last_two),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// Clears the map, removing all key-value pairs. Keeps the allocated memory
|
|
|
|
|
/// for reuse.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut a = HashMap::new();
|
|
|
|
|
/// a.insert(1u, "a");
|
|
|
|
|
/// a.clear();
|
|
|
|
|
/// assert!(a.is_empty());
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-12-16 17:45:03 -05:00
|
|
|
|
#[inline]
|
2014-10-30 13:43:24 -07:00
|
|
|
|
pub fn clear(&mut self) {
|
2014-12-16 17:45:03 -05:00
|
|
|
|
self.drain();
|
2014-10-30 13:43:24 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// Deprecated: Renamed to `get`.
|
|
|
|
|
#[deprecated = "Renamed to `get`"]
|
|
|
|
|
pub fn find(&self, k: &K) -> Option<&V> {
|
|
|
|
|
self.get(k)
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// Returns a reference to the value corresponding to the key.
|
|
|
|
|
///
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// The key may be any borrowed form of the map's key type, but
|
|
|
|
|
/// `Hash` and `Eq` on the borrowed form *must* match those for
|
|
|
|
|
/// the key type.
|
|
|
|
|
///
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert(1u, "a");
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// assert_eq!(map.get(&1), Some(&"a"));
|
|
|
|
|
/// assert_eq!(map.get(&2), None);
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
pub fn get<Sized? Q>(&self, k: &Q) -> Option<&V>
|
|
|
|
|
where Q: Hash<S> + Eq + BorrowFrom<K>
|
|
|
|
|
{
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.search(k).map(|bucket| bucket.into_refs().1)
|
2014-10-30 13:43:24 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Returns true if the map contains a value for the specified key.
|
|
|
|
|
///
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// The key may be any borrowed form of the map's key type, but
|
|
|
|
|
/// `Hash` and `Eq` on the borrowed form *must* match those for
|
|
|
|
|
/// the key type.
|
|
|
|
|
///
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert(1u, "a");
|
|
|
|
|
/// assert_eq!(map.contains_key(&1), true);
|
|
|
|
|
/// assert_eq!(map.contains_key(&2), false);
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
pub fn contains_key<Sized? Q>(&self, k: &Q) -> bool
|
|
|
|
|
where Q: Hash<S> + Eq + BorrowFrom<K>
|
|
|
|
|
{
|
2014-10-30 13:43:24 -07:00
|
|
|
|
self.search(k).is_some()
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// Deprecated: Renamed to `get_mut`.
|
|
|
|
|
#[deprecated = "Renamed to `get_mut`"]
|
|
|
|
|
pub fn find_mut(&mut self, k: &K) -> Option<&mut V> {
|
|
|
|
|
self.get_mut(k)
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// Returns a mutable reference to the value corresponding to the key.
|
|
|
|
|
///
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// The key may be any borrowed form of the map's key type, but
|
|
|
|
|
/// `Hash` and `Eq` on the borrowed form *must* match those for
|
|
|
|
|
/// the key type.
|
|
|
|
|
///
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert(1u, "a");
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// match map.get_mut(&1) {
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// Some(x) => *x = "b",
|
|
|
|
|
/// None => (),
|
|
|
|
|
/// }
|
|
|
|
|
/// assert_eq!(map[1], "b");
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
pub fn get_mut<Sized? Q>(&mut self, k: &Q) -> Option<&mut V>
|
|
|
|
|
where Q: Hash<S> + Eq + BorrowFrom<K>
|
|
|
|
|
{
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.search_mut(k).map(|bucket| bucket.into_mut_refs().1)
|
2014-10-30 13:43:24 -07:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// Deprecated: Renamed to `insert`.
|
|
|
|
|
#[deprecated = "Renamed to `insert`"]
|
|
|
|
|
pub fn swap(&mut self, k: K, v: V) -> Option<V> {
|
|
|
|
|
self.insert(k, v)
|
2014-10-30 13:43:24 -07:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Inserts a key-value pair from the map. If the key already had a value
|
|
|
|
|
/// present in the map, that value is returned. Otherwise, `None` is returned.
|
|
|
|
|
///
|
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// assert_eq!(map.insert(37u, "a"), None);
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// assert_eq!(map.is_empty(), false);
|
|
|
|
|
///
|
|
|
|
|
/// map.insert(37, "b");
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// assert_eq!(map.insert(37, "c"), Some("b"));
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// assert_eq!(map[37], "c");
|
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
|
|
|
|
pub fn insert(&mut self, k: K, v: V) -> Option<V> {
|
2014-10-30 13:43:24 -07:00
|
|
|
|
let hash = self.make_hash(&k);
|
2014-11-17 14:23:21 +01:00
|
|
|
|
self.reserve(1);
|
2014-10-30 13:43:24 -07:00
|
|
|
|
|
|
|
|
|
let mut retval = None;
|
|
|
|
|
self.insert_or_replace_with(hash, k, v, |_, val_ref, val| {
|
|
|
|
|
retval = Some(replace(val_ref, val));
|
|
|
|
|
});
|
|
|
|
|
retval
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// Deprecated: Renamed to `remove`.
|
|
|
|
|
#[deprecated = "Renamed to `remove`"]
|
|
|
|
|
pub fn pop(&mut self, k: &K) -> Option<V> {
|
|
|
|
|
self.remove(k)
|
|
|
|
|
}
|
|
|
|
|
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// Removes a key from the map, returning the value at the key if the key
|
|
|
|
|
/// was previously in the map.
|
|
|
|
|
///
|
2014-11-12 14:55:51 -08:00
|
|
|
|
/// The key may be any borrowed form of the map's key type, but
|
|
|
|
|
/// `Hash` and `Eq` on the borrowed form *must* match those for
|
|
|
|
|
/// the key type.
|
|
|
|
|
///
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// # Example
|
|
|
|
|
///
|
|
|
|
|
/// ```
|
|
|
|
|
/// use std::collections::HashMap;
|
|
|
|
|
///
|
|
|
|
|
/// let mut map = HashMap::new();
|
|
|
|
|
/// map.insert(1u, "a");
|
2014-11-06 12:24:47 -05:00
|
|
|
|
/// assert_eq!(map.remove(&1), Some("a"));
|
|
|
|
|
/// assert_eq!(map.remove(&1), None);
|
2014-10-30 13:43:24 -07:00
|
|
|
|
/// ```
|
2014-11-06 12:24:47 -05:00
|
|
|
|
#[unstable = "matches collection reform specification, waiting for dust to settle"]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
pub fn remove<Sized? Q>(&mut self, k: &Q) -> Option<V>
|
|
|
|
|
where Q: Hash<S> + Eq + BorrowFrom<K>
|
|
|
|
|
{
|
2014-10-30 13:43:24 -07:00
|
|
|
|
if self.table.size() == 0 {
|
|
|
|
|
return None
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.search_mut(k).map(|bucket| pop_internal(bucket).1)
|
2014-10-30 13:43:24 -07:00
|
|
|
|
}
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
fn search_entry_hashed<'a, K: Eq, V>(table: &'a mut RawTable<K,V>, hash: SafeHash, k: K)
|
|
|
|
|
-> Entry<'a, K, V> {
|
|
|
|
|
// Worst case, we'll find one empty bucket among `size + 1` buckets.
|
|
|
|
|
let size = table.size();
|
2014-12-16 16:00:37 -05:00
|
|
|
|
let mut probe = Bucket::new(table, hash);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
let ib = probe.index();
|
|
|
|
|
|
|
|
|
|
loop {
|
|
|
|
|
let bucket = match probe.peek() {
|
|
|
|
|
Empty(bucket) => {
|
|
|
|
|
// Found a hole!
|
|
|
|
|
return Vacant(VacantEntry {
|
|
|
|
|
hash: hash,
|
|
|
|
|
key: k,
|
|
|
|
|
elem: NoElem(bucket),
|
|
|
|
|
});
|
|
|
|
|
},
|
|
|
|
|
Full(bucket) => bucket
|
|
|
|
|
};
|
|
|
|
|
|
2014-12-16 16:00:37 -05:00
|
|
|
|
// hash matches?
|
2014-09-18 17:05:22 -04:00
|
|
|
|
if bucket.hash() == hash {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
// key matches?
|
|
|
|
|
if k == *bucket.read().0 {
|
2014-09-18 17:05:22 -04:00
|
|
|
|
return Occupied(OccupiedEntry{
|
|
|
|
|
elem: bucket,
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let robin_ib = bucket.index() as int - bucket.distance() as int;
|
|
|
|
|
|
|
|
|
|
if (ib as int) < robin_ib {
|
|
|
|
|
// Found a luckier bucket than me. Better steal his spot.
|
|
|
|
|
return Vacant(VacantEntry {
|
|
|
|
|
hash: hash,
|
|
|
|
|
key: k,
|
|
|
|
|
elem: NeqElem(bucket, robin_ib as uint),
|
|
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
probe = bucket.next();
|
|
|
|
|
assert!(probe.index() != ib + size + 1);
|
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2013-07-16 03:55:52 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V: Clone, S, H: Hasher<S>> HashMap<K, V, H> {
|
2014-11-07 14:06:18 -05:00
|
|
|
|
/// Deprecated: Use `map.get(k).cloned()`.
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-11-07 14:06:18 -05:00
|
|
|
|
/// Return a copy of the value corresponding to the key.
|
|
|
|
|
#[deprecated = "Use `map.get(k).cloned()`"]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
pub fn find_copy(&self, k: &K) -> Option<V> {
|
2014-11-07 14:06:18 -05:00
|
|
|
|
self.get(k).cloned()
|
2013-07-16 03:55:52 +02:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-07 14:06:18 -05:00
|
|
|
|
/// Deprecated: Use `map[k].clone()`.
|
2014-07-14 19:29:29 +02:00
|
|
|
|
///
|
2014-11-07 14:06:18 -05:00
|
|
|
|
/// Return a copy of the value corresponding to the key.
|
|
|
|
|
#[deprecated = "Use `map[k].clone()`"]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
pub fn get_copy(&self, k: &K) -> V {
|
2014-09-17 10:47:05 -07:00
|
|
|
|
self[*k].clone()
|
2013-07-16 03:55:52 +02:00
|
|
|
|
}
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V: PartialEq, S, H: Hasher<S>> PartialEq for HashMap<K, V, H> {
|
|
|
|
|
fn eq(&self, other: &HashMap<K, V, H>) -> bool {
|
2014-07-17 20:40:39 +02:00
|
|
|
|
if self.len() != other.len() { return false; }
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
self.iter().all(|(key, value)|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
other.get(key).map_or(false, |v| *value == *v)
|
2014-07-15 21:58:35 +01:00
|
|
|
|
)
|
2014-07-17 20:40:39 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V: Eq, S, H: Hasher<S>> Eq for HashMap<K, V, H> {}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S> + Show, V: Show, S, H: Hasher<S>> Show for HashMap<K, V, H> {
|
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
|
try!(write!(f, "{{"));
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
for (i, (k, v)) in self.iter().enumerate() {
|
|
|
|
|
if i != 0 { try!(write!(f, ", ")); }
|
|
|
|
|
try!(write!(f, "{}: {}", *k, *v));
|
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
write!(f, "}}")
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-12-15 20:04:52 -08:00
|
|
|
|
#[stable]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V, S, H: Hasher<S> + Default> Default for HashMap<K, V, H> {
|
2014-12-15 20:04:52 -08:00
|
|
|
|
#[stable]
|
2014-07-16 00:39:32 +01:00
|
|
|
|
fn default() -> HashMap<K, V, H> {
|
|
|
|
|
HashMap::with_hasher(Default::default())
|
2014-07-17 20:40:39 +02:00
|
|
|
|
}
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
impl<K: Hash<S> + Eq, Sized? Q, V, S, H: Hasher<S>> Index<Q, V> for HashMap<K, V, H>
|
|
|
|
|
where Q: BorrowFrom<K> + Hash<S> + Eq
|
|
|
|
|
{
|
2014-07-16 00:39:32 +01:00
|
|
|
|
#[inline]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
fn index<'a>(&'a self, index: &Q) -> &'a V {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
self.get(index).expect("no entry found for key")
|
2014-07-17 20:40:39 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-12 14:55:51 -08:00
|
|
|
|
impl<K: Hash<S> + Eq, Sized? Q, V, S, H: Hasher<S>> IndexMut<Q, V> for HashMap<K, V, H>
|
|
|
|
|
where Q: BorrowFrom<K> + Hash<S> + Eq
|
|
|
|
|
{
|
2014-07-16 00:39:32 +01:00
|
|
|
|
#[inline]
|
2014-11-12 14:55:51 -08:00
|
|
|
|
fn index_mut<'a>(&'a mut self, index: &Q) -> &'a mut V {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.get_mut(index).expect("no entry found for key")
|
2014-07-16 00:39:32 +01:00
|
|
|
|
}
|
2014-10-23 08:42:21 -07:00
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// HashMap iterator
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub struct Entries<'a, K: 'a, V: 'a> {
|
|
|
|
|
inner: table::Entries<'a, K, V>
|
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// HashMap mutable values iterator
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub struct MutEntries<'a, K: 'a, V: 'a> {
|
|
|
|
|
inner: table::MutEntries<'a, K, V>
|
|
|
|
|
}
|
2014-07-17 20:40:39 +02:00
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
/// HashMap move iterator
|
2014-07-15 21:58:35 +01:00
|
|
|
|
pub struct MoveEntries<K, V> {
|
2014-12-02 14:28:35 -05:00
|
|
|
|
inner: iter::Map<
|
|
|
|
|
(SafeHash, K, V),
|
|
|
|
|
(K, V),
|
|
|
|
|
table::MoveEntries<K, V>,
|
|
|
|
|
fn((SafeHash, K, V)) -> (K, V),
|
|
|
|
|
>
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-12-12 01:02:19 -06:00
|
|
|
|
/// HashMap keys iterator
|
|
|
|
|
pub struct Keys<'a, K: 'a, V: 'a> {
|
|
|
|
|
inner: Map<(&'a K, &'a V), &'a K, Entries<'a, K, V>, fn((&'a K, &'a V)) -> &'a K>
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// HashMap values iterator
|
|
|
|
|
pub struct Values<'a, K: 'a, V: 'a> {
|
|
|
|
|
inner: Map<(&'a K, &'a V), &'a V, Entries<'a, K, V>, fn((&'a K, &'a V)) -> &'a V>
|
|
|
|
|
}
|
|
|
|
|
|
2014-12-16 17:45:03 -05:00
|
|
|
|
/// HashMap drain iterator
|
|
|
|
|
pub struct Drain<'a, K: 'a, V: 'a> {
|
|
|
|
|
inner: iter::Map<
|
|
|
|
|
(SafeHash, K, V),
|
|
|
|
|
(K, V),
|
|
|
|
|
table::Drain<'a, K, V>,
|
|
|
|
|
fn((SafeHash, K, V)) -> (K, V),
|
|
|
|
|
>
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-18 17:05:22 -04:00
|
|
|
|
/// A view into a single occupied location in a HashMap
|
|
|
|
|
pub struct OccupiedEntry<'a, K:'a, V:'a> {
|
|
|
|
|
elem: FullBucket<K, V, &'a mut RawTable<K, V>>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A view into a single empty location in a HashMap
|
|
|
|
|
pub struct VacantEntry<'a, K:'a, V:'a> {
|
|
|
|
|
hash: SafeHash,
|
|
|
|
|
key: K,
|
|
|
|
|
elem: VacantEntryState<K,V, &'a mut RawTable<K, V>>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// A view into a single location in a map, which may be vacant or occupied
|
|
|
|
|
pub enum Entry<'a, K:'a, V:'a> {
|
|
|
|
|
/// An occupied Entry
|
|
|
|
|
Occupied(OccupiedEntry<'a, K, V>),
|
|
|
|
|
/// A vacant Entry
|
|
|
|
|
Vacant(VacantEntry<'a, K, V>),
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Possible states of a VacantEntry
|
|
|
|
|
enum VacantEntryState<K, V, M> {
|
|
|
|
|
/// The index is occupied, but the key to insert has precedence,
|
|
|
|
|
/// and will kick the current one out on insertion
|
|
|
|
|
NeqElem(FullBucket<K, V, M>, uint),
|
|
|
|
|
/// The index is genuinely vacant
|
|
|
|
|
NoElem(EmptyBucket<K, V, M>),
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
impl<'a, K, V> Iterator<(&'a K, &'a V)> for Entries<'a, K, V> {
|
2014-12-12 01:02:19 -06:00
|
|
|
|
#[inline] fn next(&mut self) -> Option<(&'a K, &'a V)> { self.inner.next() }
|
|
|
|
|
#[inline] fn size_hint(&self) -> (uint, Option<uint>) { self.inner.size_hint() }
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a, K, V> Iterator<(&'a K, &'a mut V)> for MutEntries<'a, K, V> {
|
2014-12-12 01:02:19 -06:00
|
|
|
|
#[inline] fn next(&mut self) -> Option<(&'a K, &'a mut V)> { self.inner.next() }
|
|
|
|
|
#[inline] fn size_hint(&self) -> (uint, Option<uint>) { self.inner.size_hint() }
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<K, V> Iterator<(K, V)> for MoveEntries<K, V> {
|
2014-12-12 01:02:19 -06:00
|
|
|
|
#[inline] fn next(&mut self) -> Option<(K, V)> { self.inner.next() }
|
|
|
|
|
#[inline] fn size_hint(&self) -> (uint, Option<uint>) { self.inner.size_hint() }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a, K, V> Iterator<&'a K> for Keys<'a, K, V> {
|
|
|
|
|
#[inline] fn next(&mut self) -> Option<(&'a K)> { self.inner.next() }
|
|
|
|
|
#[inline] fn size_hint(&self) -> (uint, Option<uint>) { self.inner.size_hint() }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a, K, V> Iterator<&'a V> for Values<'a, K, V> {
|
|
|
|
|
#[inline] fn next(&mut self) -> Option<(&'a V)> { self.inner.next() }
|
|
|
|
|
#[inline] fn size_hint(&self) -> (uint, Option<uint>) { self.inner.size_hint() }
|
2014-07-15 21:58:35 +01:00
|
|
|
|
}
|
2014-05-28 09:24:28 -07:00
|
|
|
|
|
2014-12-16 17:45:03 -05:00
|
|
|
|
impl<'a, K: 'a, V: 'a> Iterator<(K, V)> for Drain<'a, K, V> {
|
|
|
|
|
#[inline]
|
|
|
|
|
fn next(&mut self) -> Option<(K, V)> {
|
|
|
|
|
self.inner.next()
|
|
|
|
|
}
|
|
|
|
|
#[inline]
|
|
|
|
|
fn size_hint(&self) -> (uint, Option<uint>) {
|
|
|
|
|
self.inner.size_hint()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-09-18 17:05:22 -04:00
|
|
|
|
impl<'a, K, V> OccupiedEntry<'a, K, V> {
|
|
|
|
|
/// Gets a reference to the value in the entry
|
|
|
|
|
pub fn get(&self) -> &V {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.elem.read().1
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Gets a mutable reference to the value in the entry
|
|
|
|
|
pub fn get_mut(&mut self) -> &mut V {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.elem.read_mut().1
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Converts the OccupiedEntry into a mutable reference to the value in the entry
|
|
|
|
|
/// with a lifetime bound to the map itself
|
|
|
|
|
pub fn into_mut(self) -> &'a mut V {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
self.elem.into_mut_refs().1
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Sets the value of the entry, and returns the entry's old value
|
|
|
|
|
pub fn set(&mut self, mut value: V) -> V {
|
|
|
|
|
let old_value = self.get_mut();
|
|
|
|
|
mem::swap(&mut value, old_value);
|
|
|
|
|
value
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Takes the value out of the entry, and returns it
|
|
|
|
|
pub fn take(self) -> V {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
pop_internal(self.elem).1
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl<'a, K, V> VacantEntry<'a, K, V> {
|
|
|
|
|
/// Sets the value of the entry with the VacantEntry's key,
|
|
|
|
|
/// and returns a mutable reference to it
|
|
|
|
|
pub fn set(self, value: V) -> &'a mut V {
|
|
|
|
|
match self.elem {
|
|
|
|
|
NeqElem(bucket, ib) => {
|
|
|
|
|
robin_hood(bucket, ib, self.hash, self.key, value)
|
|
|
|
|
}
|
|
|
|
|
NoElem(bucket) => {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
bucket.put(self.hash, self.key, value).into_mut_refs().1
|
2014-09-18 17:05:22 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-16 00:39:32 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V, S, H: Hasher<S> + Default> FromIterator<(K, V)> for HashMap<K, V, H> {
|
|
|
|
|
fn from_iter<T: Iterator<(K, V)>>(iter: T) -> HashMap<K, V, H> {
|
2014-12-16 16:00:37 -05:00
|
|
|
|
let lower = iter.size_hint().0;
|
2014-07-16 00:39:32 +01:00
|
|
|
|
let mut map = HashMap::with_capacity_and_hasher(lower, Default::default());
|
|
|
|
|
map.extend(iter);
|
|
|
|
|
map
|
2013-07-30 02:17:17 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2013-07-14 12:20:48 -04:00
|
|
|
|
|
2014-11-08 01:39:39 +01:00
|
|
|
|
impl<K: Eq + Hash<S>, V, S, H: Hasher<S> + Default> Extend<(K, V)> for HashMap<K, V, H> {
|
2014-07-16 00:39:32 +01:00
|
|
|
|
fn extend<T: Iterator<(K, V)>>(&mut self, mut iter: T) {
|
|
|
|
|
for (k, v) in iter {
|
|
|
|
|
self.insert(k, v);
|
2013-07-14 12:20:48 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-07-14 13:18:50 -04:00
|
|
|
|
|
2013-04-30 13:02:29 +02:00
|
|
|
|
#[cfg(test)]
|
2013-04-03 08:45:14 -04:00
|
|
|
|
mod test_map {
|
2014-05-29 19:03:06 -07:00
|
|
|
|
use prelude::*;
|
|
|
|
|
|
2014-02-28 22:23:53 -05:00
|
|
|
|
use super::HashMap;
|
2014-12-13 11:15:18 -05:00
|
|
|
|
use super::Entry::{Occupied, Vacant};
|
2014-05-29 19:03:06 -07:00
|
|
|
|
use hash;
|
2014-12-19 14:02:22 +02:00
|
|
|
|
use iter::{range_inclusive, range_step_inclusive};
|
2014-05-29 19:03:06 -07:00
|
|
|
|
use cell::RefCell;
|
2014-11-25 01:16:50 -05:00
|
|
|
|
use rand::{weak_rng, Rng};
|
2013-02-10 15:30:44 -08:00
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
struct KindaIntLike(int);
|
|
|
|
|
|
|
|
|
|
impl Equiv<int> for KindaIntLike {
|
|
|
|
|
fn equiv(&self, other: &int) -> bool {
|
|
|
|
|
let KindaIntLike(this) = *self;
|
|
|
|
|
this == *other
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-05-29 19:03:06 -07:00
|
|
|
|
impl<S: hash::Writer> hash::Hash<S> for KindaIntLike {
|
2014-04-10 20:55:34 +10:00
|
|
|
|
fn hash(&self, state: &mut S) {
|
|
|
|
|
let KindaIntLike(this) = *self;
|
|
|
|
|
this.hash(state)
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-04-02 16:54:22 -07:00
|
|
|
|
|
2013-05-30 12:03:11 -05:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_create_capacity_zero() {
|
|
|
|
|
let mut m = HashMap::with_capacity(0);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 1i).is_none());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
assert!(m.contains_key(&1));
|
|
|
|
|
assert!(!m.contains_key(&0));
|
2013-05-30 12:03:11 -05:00
|
|
|
|
}
|
|
|
|
|
|
2012-10-08 07:05:01 -07:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_insert() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(m.len(), 0);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 2i).is_none());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(m.len(), 1);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(2i, 4i).is_none());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(m.len(), 2);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
|
|
|
|
assert_eq!(*m.get(&2).unwrap(), 4);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-14 09:18:10 -08:00
|
|
|
|
thread_local! { static DROP_VECTOR: RefCell<Vec<int>> = RefCell::new(Vec::new()) }
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-05-31 10:43:52 -07:00
|
|
|
|
#[deriving(Hash, PartialEq, Eq)]
|
2014-02-28 22:23:53 -05:00
|
|
|
|
struct Dropable {
|
2014-04-01 20:39:26 -07:00
|
|
|
|
k: uint
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Dropable {
|
2014-04-01 20:39:26 -07:00
|
|
|
|
fn new(k: uint) -> Dropable {
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|slot| {
|
|
|
|
|
slot.borrow_mut()[k] += 1;
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
Dropable { k: k }
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Drop for Dropable {
|
|
|
|
|
fn drop(&mut self) {
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|slot| {
|
|
|
|
|
slot.borrow_mut()[self.k] -= 1;
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-07-09 21:21:46 +01:00
|
|
|
|
impl Clone for Dropable {
|
|
|
|
|
fn clone(&self) -> Dropable {
|
|
|
|
|
Dropable::new(self.k)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-02-28 22:23:53 -05:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_drops() {
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|slot| {
|
|
|
|
|
*slot.borrow_mut() = Vec::from_elem(200, 0i);
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 0);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-04-01 20:39:26 -07:00
|
|
|
|
for i in range(0u, 100) {
|
2014-02-28 22:23:53 -05:00
|
|
|
|
let d1 = Dropable::new(i);
|
|
|
|
|
let d2 = Dropable::new(i+100);
|
|
|
|
|
m.insert(d1, d2);
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 1);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-04-01 20:39:26 -07:00
|
|
|
|
for i in range(0u, 50) {
|
2014-02-28 22:23:53 -05:00
|
|
|
|
let k = Dropable::new(i);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
let v = m.remove(&k);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
assert!(v.is_some());
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 1);
|
|
|
|
|
assert_eq!(v.borrow()[i+100], 1);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 50) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 0);
|
|
|
|
|
assert_eq!(v.borrow()[i+100], 0);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
for i in range(50u, 100) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 1);
|
|
|
|
|
assert_eq!(v.borrow()[i+100], 1);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 0);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
2014-07-09 21:21:46 +01:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_move_iter_drops() {
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
*v.borrow_mut() = Vec::from_elem(200, 0i);
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
|
|
|
|
let hm = {
|
|
|
|
|
let mut hm = HashMap::new();
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 0);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
|
|
|
|
for i in range(0u, 100) {
|
|
|
|
|
let d1 = Dropable::new(i);
|
|
|
|
|
let d2 = Dropable::new(i+100);
|
|
|
|
|
hm.insert(d1, d2);
|
|
|
|
|
}
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 1);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
|
|
|
|
hm
|
|
|
|
|
};
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// By the way, ensure that cloning doesn't screw up the dropping.
|
2014-07-09 21:21:46 +01:00
|
|
|
|
drop(hm.clone());
|
|
|
|
|
|
|
|
|
|
{
|
2014-09-14 20:27:36 -07:00
|
|
|
|
let mut half = hm.into_iter().take(50);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 1);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
|
|
|
|
for _ in half {}
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
let nk = range(0u, 100).filter(|&i| {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
v.borrow()[i] == 1
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}).count();
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
let nv = range(0u, 100).filter(|&i| {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
v.borrow()[i+100] == 1
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}).count();
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
assert_eq!(nk, 50);
|
|
|
|
|
assert_eq!(nv, 50);
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
};
|
|
|
|
|
|
2014-11-14 14:20:57 -08:00
|
|
|
|
DROP_VECTOR.with(|v| {
|
|
|
|
|
for i in range(0u, 200) {
|
2014-11-27 14:43:55 -05:00
|
|
|
|
assert_eq!(v.borrow()[i], 0);
|
2014-11-14 14:20:57 -08:00
|
|
|
|
}
|
|
|
|
|
});
|
2014-07-09 21:21:46 +01:00
|
|
|
|
}
|
|
|
|
|
|
2014-02-28 22:23:53 -05:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_empty_pop() {
|
|
|
|
|
let mut m: HashMap<int, bool> = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(m.remove(&0), None);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_lots_of_insertions() {
|
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
|
|
|
|
|
// Try this a few times to make sure we never screw up the hashmap's
|
|
|
|
|
// internal state.
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for _ in range(0i, 10) {
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert!(m.is_empty());
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_inclusive(1i, 1000) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(i, i).is_none());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
for j in range_inclusive(1, i) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
let r = m.get(&j);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(r, Some(&j));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for j in range_inclusive(i+1, 1000) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
let r = m.get(&j);
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(r, None);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_inclusive(1001i, 2000) {
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert!(!m.contains_key(&i));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// remove forwards
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_inclusive(1i, 1000) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.remove(&i).is_some());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
for j in range_inclusive(1, i) {
|
|
|
|
|
assert!(!m.contains_key(&j));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for j in range_inclusive(i+1, 1000) {
|
|
|
|
|
assert!(m.contains_key(&j));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_inclusive(1i, 1000) {
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert!(!m.contains_key(&i));
|
|
|
|
|
}
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_inclusive(1i, 1000) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(i, i).is_none());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// remove backwards
|
2014-04-21 17:58:52 -04:00
|
|
|
|
for i in range_step_inclusive(1000i, 1, -1) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.remove(&i).is_some());
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
|
|
|
|
for j in range_inclusive(i, 1000) {
|
|
|
|
|
assert!(!m.contains_key(&j));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for j in range_inclusive(1, i-1) {
|
|
|
|
|
assert!(m.contains_key(&j));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_find_mut() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 12i).is_none());
|
|
|
|
|
assert!(m.insert(2i, 8i).is_none());
|
|
|
|
|
assert!(m.insert(5i, 14i).is_none());
|
2013-04-03 08:45:14 -04:00
|
|
|
|
let new = 100;
|
2014-11-06 12:24:47 -05:00
|
|
|
|
match m.get_mut(&5) {
|
2014-10-09 15:17:22 -04:00
|
|
|
|
None => panic!(), Some(x) => *x = new
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(m.get(&5), Some(&new));
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-24 19:07:36 -04:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_insert_overwrite() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 2i).is_none());
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
|
|
|
|
assert!(!m.insert(1i, 3i).is_none());
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 3);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_insert_conflicts() {
|
2013-08-01 18:11:25 +10:00
|
|
|
|
let mut m = HashMap::with_capacity(4);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 2i).is_none());
|
|
|
|
|
assert!(m.insert(5i, 3i).is_none());
|
|
|
|
|
assert!(m.insert(9i, 4i).is_none());
|
|
|
|
|
assert_eq!(*m.get(&9).unwrap(), 4);
|
|
|
|
|
assert_eq!(*m.get(&5).unwrap(), 3);
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_conflict_remove() {
|
2013-08-01 18:11:25 +10:00
|
|
|
|
let mut m = HashMap::with_capacity(4);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 2i).is_none());
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
|
|
|
|
assert!(m.insert(5, 3).is_none());
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
|
|
|
|
assert_eq!(*m.get(&5).unwrap(), 3);
|
|
|
|
|
assert!(m.insert(9, 4).is_none());
|
|
|
|
|
assert_eq!(*m.get(&1).unwrap(), 2);
|
|
|
|
|
assert_eq!(*m.get(&5).unwrap(), 3);
|
|
|
|
|
assert_eq!(*m.get(&9).unwrap(), 4);
|
|
|
|
|
assert!(m.remove(&1).is_some());
|
|
|
|
|
assert_eq!(*m.get(&9).unwrap(), 4);
|
|
|
|
|
assert_eq!(*m.get(&5).unwrap(), 3);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_is_empty() {
|
2013-08-01 18:11:25 +10:00
|
|
|
|
let mut m = HashMap::with_capacity(4);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(1i, 2i).is_none());
|
2013-04-03 08:45:14 -04:00
|
|
|
|
assert!(!m.is_empty());
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.remove(&1).is_some());
|
2013-04-03 08:45:14 -04:00
|
|
|
|
assert!(m.is_empty());
|
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_pop() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m.insert(1i, 2i);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(m.remove(&1), Some(2));
|
|
|
|
|
assert_eq!(m.remove(&1), None);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
2014-04-02 16:54:22 -07:00
|
|
|
|
#[test]
|
2014-04-10 20:55:34 +10:00
|
|
|
|
#[allow(experimental)]
|
2014-04-02 16:54:22 -07:00
|
|
|
|
fn test_pop_equiv() {
|
|
|
|
|
let mut m = HashMap::new();
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m.insert(1i, 2i);
|
2014-04-10 20:55:34 +10:00
|
|
|
|
assert_eq!(m.pop_equiv(&KindaIntLike(1)), Some(2));
|
|
|
|
|
assert_eq!(m.pop_equiv(&KindaIntLike(1)), None);
|
2014-04-02 16:54:22 -07:00
|
|
|
|
}
|
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_iterate() {
|
2013-08-01 18:11:25 +10:00
|
|
|
|
let mut m = HashMap::with_capacity(4);
|
2013-08-03 12:45:23 -04:00
|
|
|
|
for i in range(0u, 32) {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.insert(i, i*2).is_none());
|
2012-07-31 17:11:57 -04:00
|
|
|
|
}
|
2014-02-28 22:23:53 -05:00
|
|
|
|
assert_eq!(m.len(), 32);
|
|
|
|
|
|
2014-05-14 00:44:04 +09:00
|
|
|
|
let mut observed: u32 = 0;
|
2014-02-28 22:23:53 -05:00
|
|
|
|
|
2013-08-03 12:45:23 -04:00
|
|
|
|
for (k, v) in m.iter() {
|
2013-05-18 22:02:45 -04:00
|
|
|
|
assert_eq!(*v, *k * 2);
|
2014-02-28 01:23:06 -08:00
|
|
|
|
observed |= 1 << *k;
|
2012-09-14 15:19:54 -07:00
|
|
|
|
}
|
2013-05-18 22:02:45 -04:00
|
|
|
|
assert_eq!(observed, 0xFFFF_FFFF);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2012-12-10 09:00:52 -08:00
|
|
|
|
|
2013-12-31 19:09:18 +00:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_keys() {
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let vec = vec![(1i, 'a'), (2i, 'b'), (3i, 'c')];
|
2014-09-14 20:27:36 -07:00
|
|
|
|
let map = vec.into_iter().collect::<HashMap<int, char>>();
|
2014-04-05 16:45:42 +11:00
|
|
|
|
let keys = map.keys().map(|&k| k).collect::<Vec<int>>();
|
2013-12-31 19:09:18 +00:00
|
|
|
|
assert_eq!(keys.len(), 3);
|
|
|
|
|
assert!(keys.contains(&1));
|
|
|
|
|
assert!(keys.contains(&2));
|
|
|
|
|
assert!(keys.contains(&3));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_values() {
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let vec = vec![(1i, 'a'), (2i, 'b'), (3i, 'c')];
|
2014-09-14 20:27:36 -07:00
|
|
|
|
let map = vec.into_iter().collect::<HashMap<int, char>>();
|
2014-04-05 16:45:42 +11:00
|
|
|
|
let values = map.values().map(|&v| v).collect::<Vec<char>>();
|
2013-12-31 19:09:18 +00:00
|
|
|
|
assert_eq!(values.len(), 3);
|
|
|
|
|
assert!(values.contains(&'a'));
|
|
|
|
|
assert!(values.contains(&'b'));
|
|
|
|
|
assert!(values.contains(&'c'));
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_find() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.get(&1i).is_none());
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m.insert(1i, 2i);
|
2014-11-06 12:24:47 -05:00
|
|
|
|
match m.get(&1) {
|
2014-10-09 15:17:22 -04:00
|
|
|
|
None => panic!(),
|
2014-04-21 19:07:59 -04:00
|
|
|
|
Some(v) => assert_eq!(*v, 2)
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
2012-12-10 09:00:52 -08:00
|
|
|
|
|
2014-07-09 21:21:46 +01:00
|
|
|
|
#[test]
|
2014-11-07 14:35:18 -05:00
|
|
|
|
#[allow(deprecated)]
|
2014-07-09 21:21:46 +01:00
|
|
|
|
fn test_find_copy() {
|
|
|
|
|
let mut m = HashMap::new();
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert!(m.get(&1i).is_none());
|
2014-07-09 21:21:46 +01:00
|
|
|
|
|
|
|
|
|
for i in range(1i, 10000) {
|
|
|
|
|
m.insert(i, i + 7);
|
|
|
|
|
match m.find_copy(&i) {
|
2014-10-09 15:17:22 -04:00
|
|
|
|
None => panic!(),
|
2014-07-09 21:21:46 +01:00
|
|
|
|
Some(v) => assert_eq!(v, i + 7)
|
|
|
|
|
}
|
|
|
|
|
for j in range(1i, i/100) {
|
|
|
|
|
match m.find_copy(&j) {
|
2014-10-09 15:17:22 -04:00
|
|
|
|
None => panic!(),
|
2014-07-09 21:21:46 +01:00
|
|
|
|
Some(v) => assert_eq!(v, j + 7)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_eq() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m1 = HashMap::new();
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m1.insert(1i, 2i);
|
|
|
|
|
m1.insert(2i, 3i);
|
|
|
|
|
m1.insert(3i, 4i);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m2 = HashMap::new();
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m2.insert(1i, 2i);
|
|
|
|
|
m2.insert(2i, 3i);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
assert!(m1 != m2);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
m2.insert(3i, 4i);
|
2012-12-30 19:35:03 +01:00
|
|
|
|
|
2013-05-18 22:02:45 -04:00
|
|
|
|
assert_eq!(m1, m2);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|
2012-12-30 19:35:03 +01:00
|
|
|
|
|
2014-05-26 15:20:39 -07:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_show() {
|
|
|
|
|
let mut map: HashMap<int, int> = HashMap::new();
|
|
|
|
|
let empty: HashMap<int, int> = HashMap::new();
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
map.insert(1i, 2i);
|
|
|
|
|
map.insert(3i, 4i);
|
2014-05-26 15:20:39 -07:00
|
|
|
|
|
|
|
|
|
let map_str = format!("{}", map);
|
|
|
|
|
|
2014-11-27 19:45:47 -05:00
|
|
|
|
assert!(map_str == "{1: 2, 3: 4}" || map_str == "{3: 4, 1: 2}");
|
|
|
|
|
assert_eq!(format!("{}", empty), "{}");
|
2014-05-26 15:20:39 -07:00
|
|
|
|
}
|
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
#[test]
|
2013-04-16 01:08:52 +10:00
|
|
|
|
fn test_expand() {
|
2013-04-03 09:28:36 -04:00
|
|
|
|
let mut m = HashMap::new();
|
2012-12-30 19:35:03 +01:00
|
|
|
|
|
2013-05-18 22:02:45 -04:00
|
|
|
|
assert_eq!(m.len(), 0);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
assert!(m.is_empty());
|
2012-12-30 19:35:03 +01:00
|
|
|
|
|
2013-04-03 08:45:14 -04:00
|
|
|
|
let mut i = 0u;
|
2014-06-02 10:26:02 +02:00
|
|
|
|
let old_cap = m.table.capacity();
|
|
|
|
|
while old_cap == m.table.capacity() {
|
2013-04-03 08:45:14 -04:00
|
|
|
|
m.insert(i, i);
|
|
|
|
|
i += 1;
|
2013-03-02 05:09:36 -05:00
|
|
|
|
}
|
2013-04-03 08:45:14 -04:00
|
|
|
|
|
2013-05-18 22:02:45 -04:00
|
|
|
|
assert_eq!(m.len(), i);
|
2013-04-03 08:45:14 -04:00
|
|
|
|
assert!(!m.is_empty());
|
2012-12-30 19:35:03 +01:00
|
|
|
|
}
|
2013-06-11 20:33:30 +10:00
|
|
|
|
|
2014-06-02 10:26:02 +02:00
|
|
|
|
#[test]
|
2014-11-17 14:23:21 +01:00
|
|
|
|
fn test_behavior_resize_policy() {
|
2014-06-02 10:26:02 +02:00
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
|
|
|
|
|
assert_eq!(m.len(), 0);
|
2014-07-09 21:21:46 +01:00
|
|
|
|
assert_eq!(m.table.capacity(), 0);
|
2014-06-02 10:26:02 +02:00
|
|
|
|
assert!(m.is_empty());
|
|
|
|
|
|
2014-07-09 21:21:46 +01:00
|
|
|
|
m.insert(0, 0);
|
|
|
|
|
m.remove(&0);
|
|
|
|
|
assert!(m.is_empty());
|
2014-06-02 10:26:02 +02:00
|
|
|
|
let initial_cap = m.table.capacity();
|
2014-11-17 14:23:21 +01:00
|
|
|
|
m.reserve(initial_cap);
|
2014-06-02 10:26:02 +02:00
|
|
|
|
let cap = m.table.capacity();
|
|
|
|
|
|
|
|
|
|
assert_eq!(cap, initial_cap * 2);
|
|
|
|
|
|
|
|
|
|
let mut i = 0u;
|
|
|
|
|
for _ in range(0, cap * 3 / 4) {
|
|
|
|
|
m.insert(i, i);
|
|
|
|
|
i += 1;
|
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// three quarters full
|
2014-06-02 10:26:02 +02:00
|
|
|
|
|
|
|
|
|
assert_eq!(m.len(), i);
|
|
|
|
|
assert_eq!(m.table.capacity(), cap);
|
|
|
|
|
|
|
|
|
|
for _ in range(0, cap / 4) {
|
|
|
|
|
m.insert(i, i);
|
|
|
|
|
i += 1;
|
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// half full
|
2014-06-02 10:26:02 +02:00
|
|
|
|
|
|
|
|
|
let new_cap = m.table.capacity();
|
|
|
|
|
assert_eq!(new_cap, cap * 2);
|
|
|
|
|
|
2014-07-15 21:58:35 +01:00
|
|
|
|
for _ in range(0, cap / 2 - 1) {
|
2014-06-02 10:26:02 +02:00
|
|
|
|
i -= 1;
|
|
|
|
|
m.remove(&i);
|
|
|
|
|
assert_eq!(m.table.capacity(), new_cap);
|
|
|
|
|
}
|
2014-07-15 21:58:35 +01:00
|
|
|
|
// A little more than one quarter full.
|
2014-11-17 14:23:21 +01:00
|
|
|
|
m.shrink_to_fit();
|
|
|
|
|
assert_eq!(m.table.capacity(), cap);
|
|
|
|
|
// again, a little more than half full
|
2014-06-02 10:26:02 +02:00
|
|
|
|
for _ in range(0, cap / 2 - 1) {
|
|
|
|
|
i -= 1;
|
|
|
|
|
m.remove(&i);
|
|
|
|
|
}
|
2014-11-17 14:23:21 +01:00
|
|
|
|
m.shrink_to_fit();
|
2014-06-02 10:26:02 +02:00
|
|
|
|
|
|
|
|
|
assert_eq!(m.len(), i);
|
|
|
|
|
assert!(!m.is_empty());
|
2014-11-17 14:23:21 +01:00
|
|
|
|
assert_eq!(m.table.capacity(), initial_cap);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_reserve_shrink_to_fit() {
|
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
m.insert(0u, 0u);
|
|
|
|
|
m.remove(&0);
|
|
|
|
|
assert!(m.capacity() >= m.len());
|
|
|
|
|
for i in range(0, 128) {
|
|
|
|
|
m.insert(i, i);
|
|
|
|
|
}
|
|
|
|
|
m.reserve(256);
|
|
|
|
|
|
|
|
|
|
let usable_cap = m.capacity();
|
|
|
|
|
for i in range(128, 128+256) {
|
|
|
|
|
m.insert(i, i);
|
|
|
|
|
assert_eq!(m.capacity(), usable_cap);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i in range(100, 128+256) {
|
|
|
|
|
assert_eq!(m.remove(&i), Some(i));
|
|
|
|
|
}
|
|
|
|
|
m.shrink_to_fit();
|
|
|
|
|
|
|
|
|
|
assert_eq!(m.len(), 100);
|
|
|
|
|
assert!(!m.is_empty());
|
|
|
|
|
assert!(m.capacity() >= m.len());
|
|
|
|
|
|
|
|
|
|
for i in range(0, 100) {
|
|
|
|
|
assert_eq!(m.remove(&i), Some(i));
|
|
|
|
|
}
|
|
|
|
|
m.shrink_to_fit();
|
|
|
|
|
m.insert(0, 0);
|
|
|
|
|
|
|
|
|
|
assert_eq!(m.len(), 1);
|
|
|
|
|
assert!(m.capacity() >= m.len());
|
|
|
|
|
assert_eq!(m.remove(&0), Some(0));
|
2014-06-02 10:26:02 +02:00
|
|
|
|
}
|
|
|
|
|
|
2013-06-11 20:33:30 +10:00
|
|
|
|
#[test]
|
|
|
|
|
fn test_find_equiv() {
|
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let (foo, bar, baz) = (1i,2i,3i);
|
2014-05-25 03:10:11 -07:00
|
|
|
|
m.insert("foo".to_string(), foo);
|
|
|
|
|
m.insert("bar".to_string(), bar);
|
|
|
|
|
m.insert("baz".to_string(), baz);
|
2013-06-11 20:33:30 +10:00
|
|
|
|
|
|
|
|
|
|
2014-11-12 15:51:51 -08:00
|
|
|
|
assert_eq!(m.get("foo"), Some(&foo));
|
|
|
|
|
assert_eq!(m.get("bar"), Some(&bar));
|
|
|
|
|
assert_eq!(m.get("baz"), Some(&baz));
|
2013-06-11 20:33:30 +10:00
|
|
|
|
|
2014-11-12 15:51:51 -08:00
|
|
|
|
assert_eq!(m.get("qux"), None);
|
2013-06-11 20:33:30 +10:00
|
|
|
|
}
|
2013-07-14 11:26:03 -04:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_from_iter() {
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let xs = [(1i, 1i), (2, 2), (3, 3), (4, 4), (5, 5), (6, 6)];
|
2013-07-14 13:18:50 -04:00
|
|
|
|
|
2013-08-09 20:09:47 -07:00
|
|
|
|
let map: HashMap<int, int> = xs.iter().map(|&x| x).collect();
|
2013-07-14 11:26:03 -04:00
|
|
|
|
|
2013-08-03 12:45:23 -04:00
|
|
|
|
for &(k, v) in xs.iter() {
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(map.get(&k), Some(&v));
|
2013-07-14 11:26:03 -04:00
|
|
|
|
}
|
|
|
|
|
}
|
2014-04-21 19:07:59 -04:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_size_hint() {
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let xs = [(1i, 1i), (2, 2), (3, 3), (4, 4), (5, 5), (6, 6)];
|
2014-04-21 19:07:59 -04:00
|
|
|
|
|
|
|
|
|
let map: HashMap<int, int> = xs.iter().map(|&x| x).collect();
|
|
|
|
|
|
|
|
|
|
let mut iter = map.iter();
|
|
|
|
|
|
|
|
|
|
for _ in iter.by_ref().take(3) {}
|
|
|
|
|
|
|
|
|
|
assert_eq!(iter.size_hint(), (3, Some(3)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_mut_size_hint() {
|
2014-04-21 17:58:52 -04:00
|
|
|
|
let xs = [(1i, 1i), (2, 2), (3, 3), (4, 4), (5, 5), (6, 6)];
|
2014-04-21 19:07:59 -04:00
|
|
|
|
|
|
|
|
|
let mut map: HashMap<int, int> = xs.iter().map(|&x| x).collect();
|
|
|
|
|
|
2014-09-14 20:27:36 -07:00
|
|
|
|
let mut iter = map.iter_mut();
|
2014-04-21 19:07:59 -04:00
|
|
|
|
|
|
|
|
|
for _ in iter.by_ref().take(3) {}
|
|
|
|
|
|
|
|
|
|
assert_eq!(iter.size_hint(), (3, Some(3)));
|
|
|
|
|
}
|
2014-08-02 18:48:44 +12:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_index() {
|
|
|
|
|
let mut map: HashMap<int, int> = HashMap::new();
|
|
|
|
|
|
|
|
|
|
map.insert(1, 2);
|
|
|
|
|
map.insert(2, 1);
|
|
|
|
|
map.insert(3, 4);
|
|
|
|
|
|
|
|
|
|
assert_eq!(map[2], 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
#[should_fail]
|
|
|
|
|
fn test_index_nonexistent() {
|
|
|
|
|
let mut map: HashMap<int, int> = HashMap::new();
|
|
|
|
|
|
|
|
|
|
map.insert(1, 2);
|
|
|
|
|
map.insert(2, 1);
|
|
|
|
|
map.insert(3, 4);
|
|
|
|
|
|
|
|
|
|
map[4];
|
|
|
|
|
}
|
2014-09-18 17:05:22 -04:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_entry(){
|
|
|
|
|
let xs = [(1i, 10i), (2, 20), (3, 30), (4, 40), (5, 50), (6, 60)];
|
|
|
|
|
|
|
|
|
|
let mut map: HashMap<int, int> = xs.iter().map(|&x| x).collect();
|
|
|
|
|
|
|
|
|
|
// Existing key (insert)
|
|
|
|
|
match map.entry(1) {
|
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
|
Occupied(mut view) => {
|
|
|
|
|
assert_eq!(view.get(), &10);
|
|
|
|
|
assert_eq!(view.set(100), 10);
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(map.get(&1).unwrap(), &100);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Existing key (update)
|
|
|
|
|
match map.entry(2) {
|
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
|
Occupied(mut view) => {
|
|
|
|
|
let v = view.get_mut();
|
|
|
|
|
let new_v = (*v) * 10;
|
|
|
|
|
*v = new_v;
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(map.get(&2).unwrap(), &200);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
|
|
|
|
|
|
// Existing key (take)
|
|
|
|
|
match map.entry(3) {
|
|
|
|
|
Vacant(_) => unreachable!(),
|
|
|
|
|
Occupied(view) => {
|
|
|
|
|
assert_eq!(view.take(), 30);
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(map.get(&3), None);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
assert_eq!(map.len(), 5);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
// Inexistent key (insert)
|
|
|
|
|
match map.entry(10) {
|
|
|
|
|
Occupied(_) => unreachable!(),
|
|
|
|
|
Vacant(view) => {
|
|
|
|
|
assert_eq!(*view.set(1000), 1000);
|
|
|
|
|
}
|
|
|
|
|
}
|
2014-11-06 12:24:47 -05:00
|
|
|
|
assert_eq!(map.get(&10).unwrap(), &1000);
|
2014-09-18 17:05:22 -04:00
|
|
|
|
assert_eq!(map.len(), 6);
|
|
|
|
|
}
|
2014-11-25 01:16:50 -05:00
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_entry_take_doesnt_corrupt() {
|
|
|
|
|
// Test for #19292
|
|
|
|
|
fn check(m: &HashMap<int, ()>) {
|
|
|
|
|
for k in m.keys() {
|
|
|
|
|
assert!(m.contains_key(k),
|
|
|
|
|
"{} is in keys() but not in the map?", k);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let mut m = HashMap::new();
|
|
|
|
|
let mut rng = weak_rng();
|
|
|
|
|
|
|
|
|
|
// Populate the map with some items.
|
|
|
|
|
for _ in range(0u, 50) {
|
|
|
|
|
let x = rng.gen_range(-10, 10);
|
|
|
|
|
m.insert(x, ());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for i in range(0u, 1000) {
|
|
|
|
|
let x = rng.gen_range(-10, 10);
|
|
|
|
|
match m.entry(x) {
|
|
|
|
|
Vacant(_) => {},
|
|
|
|
|
Occupied(e) => {
|
|
|
|
|
println!("{}: remove {}", i, x);
|
|
|
|
|
e.take();
|
|
|
|
|
},
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
check(&m);
|
|
|
|
|
}
|
|
|
|
|
}
|
2013-04-03 08:45:14 -04:00
|
|
|
|
}
|