2013-01-20 14:18:24 -05:00
|
|
|
// Copyright 2013 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.
|
|
|
|
|
2013-01-23 16:28:00 -05:00
|
|
|
//! Sendable hash maps.
|
2012-07-26 16:09:22 -07:00
|
|
|
|
|
|
|
/// Open addressing with linear probing.
|
2012-09-26 17:47:29 -07:00
|
|
|
pub mod linear {
|
2013-02-25 14:11:21 -05:00
|
|
|
use container::{Container, Mutable, Map, Set};
|
2013-03-04 19:43:14 -08:00
|
|
|
use cmp::{Eq, Equiv};
|
2013-02-25 14:11:21 -05:00
|
|
|
use hash::Hash;
|
|
|
|
use to_bytes::IterBytes;
|
2013-01-20 14:18:24 -05:00
|
|
|
use iter::BaseIter;
|
2013-01-08 19:37:25 -08:00
|
|
|
use hash::Hash;
|
2013-01-29 16:07:11 -05:00
|
|
|
use iter;
|
2013-01-08 19:37:25 -08:00
|
|
|
use option::{None, Option, Some};
|
2013-03-12 13:00:50 -07:00
|
|
|
use rand::RngUtil;
|
2012-12-23 17:41:37 -05:00
|
|
|
use rand;
|
|
|
|
use uint;
|
|
|
|
use vec;
|
|
|
|
|
2012-11-06 18:41:06 -08:00
|
|
|
const INITIAL_CAPACITY: uint = 32u; // 2^5
|
2012-09-26 17:47:29 -07:00
|
|
|
|
2013-01-28 10:46:43 -08:00
|
|
|
struct Bucket<K,V> {
|
2012-09-07 14:50:47 -07:00
|
|
|
hash: uint,
|
|
|
|
key: K,
|
|
|
|
value: V,
|
2012-08-28 11:59:31 -07:00
|
|
|
}
|
2013-01-24 22:00:58 -05:00
|
|
|
|
2013-01-28 10:46:43 -08:00
|
|
|
pub struct LinearMap<K,V> {
|
2013-03-02 05:09:36 -05:00
|
|
|
priv k0: u64,
|
|
|
|
priv k1: u64,
|
|
|
|
priv resize_at: uint,
|
|
|
|
priv size: uint,
|
|
|
|
priv buckets: ~[Option<Bucket<K, V>>],
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-02-06 00:03:12 -05:00
|
|
|
// We could rewrite FoundEntry to have type Option<&Bucket<K, V>>
|
|
|
|
// which would be nifty
|
2012-08-14 16:54:13 -07:00
|
|
|
enum SearchResult {
|
|
|
|
FoundEntry(uint), FoundHole(uint), TableFull
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-02-15 01:35:15 -05:00
|
|
|
#[inline(always)]
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn resize_at(capacity: uint) -> uint {
|
2012-07-26 16:09:22 -07:00
|
|
|
((capacity as float) * 3. / 4.) as uint
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub fn linear_map_with_capacity<K:Eq + Hash,V>(
|
2013-01-24 22:00:58 -05:00
|
|
|
initial_capacity: uint) -> LinearMap<K, V> {
|
2013-01-30 02:43:11 -05:00
|
|
|
let r = rand::task_rng();
|
2012-08-30 19:01:22 -07:00
|
|
|
linear_map_with_capacity_and_keys(r.gen_u64(), r.gen_u64(),
|
|
|
|
initial_capacity)
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pure fn linear_map_with_capacity_and_keys<K:Eq + Hash,V>(
|
2012-08-30 19:01:22 -07:00
|
|
|
k0: u64, k1: u64,
|
2013-01-24 22:00:58 -05:00
|
|
|
initial_capacity: uint) -> LinearMap<K, V> {
|
2012-08-28 11:59:31 -07:00
|
|
|
LinearMap {
|
2012-08-30 19:01:22 -07:00
|
|
|
k0: k0, k1: k1,
|
2012-07-26 16:09:22 -07:00
|
|
|
resize_at: resize_at(initial_capacity),
|
|
|
|
size: 0,
|
2013-01-24 22:00:58 -05:00
|
|
|
buckets: vec::from_fn(initial_capacity, |_| None)
|
2012-08-28 11:59:31 -07:00
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
priv impl<K:Hash + IterBytes + Eq,V> LinearMap<K, V> {
|
2012-07-26 16:09:22 -07:00
|
|
|
#[inline(always)]
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn to_bucket(&self, h: uint) -> uint {
|
2013-02-15 00:55:46 -05:00
|
|
|
// A good hash function with entropy spread over all of the
|
|
|
|
// bits is assumed. SipHash is more than good enough.
|
2012-07-26 16:09:22 -07:00
|
|
|
h % self.buckets.len()
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn next_bucket(&self, idx: uint, len_buckets: uint) -> uint {
|
2012-07-26 16:09:22 -07:00
|
|
|
let n = (idx + 1) % len_buckets;
|
2013-01-23 16:39:09 -05:00
|
|
|
debug!("next_bucket(%?, %?) = %?", idx, len_buckets, n);
|
2013-01-24 22:00:58 -05:00
|
|
|
n
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn bucket_sequence(&self, hash: uint,
|
2013-03-07 14:38:38 -08:00
|
|
|
op: &fn(uint) -> bool) -> uint {
|
2012-07-26 16:09:22 -07:00
|
|
|
let start_idx = self.to_bucket(hash);
|
|
|
|
let len_buckets = self.buckets.len();
|
|
|
|
let mut idx = start_idx;
|
|
|
|
loop {
|
|
|
|
if !op(idx) {
|
2012-08-01 17:30:05 -07:00
|
|
|
return idx;
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
idx = self.next_bucket(idx, len_buckets);
|
|
|
|
if idx == start_idx {
|
2012-08-01 17:30:05 -07:00
|
|
|
return start_idx;
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(always)]
|
2013-02-10 13:56:03 -08:00
|
|
|
pure fn bucket_for_key(&self, k: &K) -> SearchResult {
|
2012-08-30 19:01:22 -07:00
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
2013-02-10 13:56:03 -08:00
|
|
|
self.bucket_for_key_with_hash(hash, k)
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-03-04 19:43:14 -08:00
|
|
|
#[inline(always)]
|
|
|
|
pure fn bucket_for_key_equiv<Q:Hash + IterBytes + Equiv<K>>(
|
|
|
|
&self,
|
|
|
|
k: &Q)
|
|
|
|
-> SearchResult {
|
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
|
|
|
self.bucket_for_key_with_hash_equiv(hash, k)
|
|
|
|
}
|
|
|
|
|
2012-07-26 16:09:22 -07:00
|
|
|
#[inline(always)]
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn bucket_for_key_with_hash(&self,
|
2012-08-22 21:01:30 -07:00
|
|
|
hash: uint,
|
|
|
|
k: &K) -> SearchResult {
|
2012-07-26 16:09:22 -07:00
|
|
|
let _ = for self.bucket_sequence(hash) |i| {
|
2013-02-10 13:56:03 -08:00
|
|
|
match self.buckets[i] {
|
2012-09-28 13:00:07 -07:00
|
|
|
Some(ref bkt) => if bkt.hash == hash && *k == bkt.key {
|
2012-09-07 10:34:39 -07:00
|
|
|
return FoundEntry(i);
|
|
|
|
},
|
|
|
|
None => return FoundHole(i)
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
};
|
2013-01-24 22:00:58 -05:00
|
|
|
TableFull
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-03-04 19:43:14 -08:00
|
|
|
#[inline(always)]
|
|
|
|
pure fn bucket_for_key_with_hash_equiv<Q:Equiv<K>>(&self,
|
|
|
|
hash: uint,
|
|
|
|
k: &Q)
|
|
|
|
-> SearchResult {
|
|
|
|
let _ = for self.bucket_sequence(hash) |i| {
|
|
|
|
match self.buckets[i] {
|
|
|
|
Some(ref bkt) => {
|
|
|
|
if bkt.hash == hash && k.equiv(&bkt.key) {
|
|
|
|
return FoundEntry(i);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
None => return FoundHole(i)
|
|
|
|
}
|
|
|
|
};
|
|
|
|
TableFull
|
|
|
|
}
|
|
|
|
|
2013-02-15 01:35:15 -05:00
|
|
|
/// Expand the capacity of the array to the next power of two
|
|
|
|
/// and re-insert each of the existing buckets.
|
|
|
|
#[inline(always)]
|
2012-08-22 21:01:30 -07:00
|
|
|
fn expand(&mut self) {
|
2013-02-15 01:35:15 -05:00
|
|
|
let new_capacity = self.buckets.len() * 2;
|
|
|
|
self.resize(new_capacity);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Expands the capacity of the array and re-insert each of the
|
|
|
|
/// existing buckets.
|
|
|
|
fn resize(&mut self, new_capacity: uint) {
|
2012-07-26 16:09:22 -07:00
|
|
|
let old_capacity = self.buckets.len();
|
2013-02-15 01:35:15 -05:00
|
|
|
self.resize_at = resize_at(new_capacity);
|
2012-07-26 16:09:22 -07:00
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
let mut old_buckets = vec::from_fn(new_capacity, |_| None);
|
2012-07-26 16:09:22 -07:00
|
|
|
self.buckets <-> old_buckets;
|
|
|
|
|
2012-12-30 19:35:03 +01:00
|
|
|
self.size = 0;
|
2012-07-26 16:09:22 -07:00
|
|
|
for uint::range(0, old_capacity) |i| {
|
2012-08-20 12:23:37 -07:00
|
|
|
let mut bucket = None;
|
2012-07-26 16:09:22 -07:00
|
|
|
bucket <-> old_buckets[i];
|
2013-01-24 22:00:58 -05:00
|
|
|
self.insert_opt_bucket(bucket);
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
fn insert_opt_bucket(&mut self, bucket: Option<Bucket<K, V>>) {
|
|
|
|
match bucket {
|
|
|
|
Some(Bucket{hash: hash, key: key, value: value}) => {
|
|
|
|
self.insert_internal(hash, key, value);
|
2012-09-07 10:34:39 -07:00
|
|
|
}
|
|
|
|
None => {}
|
2012-08-28 11:59:31 -07:00
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-02-10 15:30:44 -08:00
|
|
|
#[inline(always)]
|
2013-03-14 11:22:51 -07:00
|
|
|
pure fn value_for_bucket(&self, idx: uint) -> &'self V {
|
2013-02-10 15:30:44 -08:00
|
|
|
match self.buckets[idx] {
|
|
|
|
Some(ref bkt) => &bkt.value,
|
2013-02-14 13:09:09 -08:00
|
|
|
None => fail!(~"LinearMap::find: internal logic error"),
|
2013-02-10 15:30:44 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-07-26 16:09:22 -07:00
|
|
|
/// Inserts the key value pair into the buckets.
|
|
|
|
/// Assumes that there will be a bucket.
|
|
|
|
/// True if there was no previous entry with that key
|
2012-10-02 11:37:37 -07:00
|
|
|
fn insert_internal(&mut self, hash: uint, k: K, v: V) -> bool {
|
2013-02-10 13:56:03 -08:00
|
|
|
match self.bucket_for_key_with_hash(hash, &k) {
|
2013-02-11 19:26:38 -08:00
|
|
|
TableFull => { fail!(~"Internal logic error"); }
|
2012-09-07 10:34:39 -07:00
|
|
|
FoundHole(idx) => {
|
|
|
|
debug!("insert fresh (%?->%?) at idx %?, hash %?",
|
|
|
|
k, v, idx, hash);
|
2013-01-24 22:00:58 -05:00
|
|
|
self.buckets[idx] = Some(Bucket{hash: hash, key: k,
|
|
|
|
value: v});
|
2012-09-07 10:34:39 -07:00
|
|
|
self.size += 1;
|
|
|
|
true
|
|
|
|
}
|
|
|
|
FoundEntry(idx) => {
|
|
|
|
debug!("insert overwrite (%?->%?) at idx %?, hash %?",
|
|
|
|
k, v, idx, hash);
|
2013-01-24 22:00:58 -05:00
|
|
|
self.buckets[idx] = Some(Bucket{hash: hash, key: k,
|
|
|
|
value: v});
|
2012-09-07 10:34:39 -07:00
|
|
|
false
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-22 21:01:30 -07:00
|
|
|
|
2012-10-08 07:05:01 -07:00
|
|
|
fn pop_internal(&mut self, hash: uint, k: &K) -> Option<V> {
|
|
|
|
// Removing from an open-addressed hashtable
|
|
|
|
// is, well, painful. The problem is that
|
|
|
|
// the entry may lie on the probe path for other
|
|
|
|
// entries, so removing it would make you think that
|
|
|
|
// those probe paths are empty.
|
|
|
|
//
|
|
|
|
// To address this we basically have to keep walking,
|
|
|
|
// re-inserting entries we find until we reach an empty
|
|
|
|
// bucket. We know we will eventually reach one because
|
|
|
|
// we insert one ourselves at the beginning (the removed
|
|
|
|
// entry).
|
|
|
|
//
|
|
|
|
// I found this explanation elucidating:
|
|
|
|
// http://www.maths.lse.ac.uk/Courses/MA407/del-hash.pdf
|
2013-02-10 13:56:03 -08:00
|
|
|
let mut idx = match self.bucket_for_key_with_hash(hash, k) {
|
2012-10-08 07:05:01 -07:00
|
|
|
TableFull | FoundHole(_) => return None,
|
|
|
|
FoundEntry(idx) => idx
|
|
|
|
};
|
|
|
|
|
|
|
|
let len_buckets = self.buckets.len();
|
|
|
|
let mut bucket = None;
|
|
|
|
self.buckets[idx] <-> bucket;
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
let value = match bucket {
|
2012-10-08 07:05:01 -07:00
|
|
|
None => None,
|
2013-01-24 22:00:58 -05:00
|
|
|
Some(bucket) => {
|
|
|
|
let Bucket{value: value, _} = bucket;
|
|
|
|
Some(value)
|
2012-10-08 07:05:01 -07:00
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-01-29 11:44:46 -08:00
|
|
|
/* re-inserting buckets may cause changes in size, so remember
|
|
|
|
what our new size is ahead of time before we start insertions */
|
2013-01-29 11:16:39 -05:00
|
|
|
let size = self.size - 1;
|
2012-10-08 07:05:01 -07:00
|
|
|
idx = self.next_bucket(idx, len_buckets);
|
|
|
|
while self.buckets[idx].is_some() {
|
|
|
|
let mut bucket = None;
|
|
|
|
bucket <-> self.buckets[idx];
|
2013-01-24 22:00:58 -05:00
|
|
|
self.insert_opt_bucket(bucket);
|
2012-10-08 07:05:01 -07:00
|
|
|
idx = self.next_bucket(idx, len_buckets);
|
|
|
|
}
|
2013-01-29 11:16:39 -05:00
|
|
|
self.size = size;
|
2012-10-08 07:05:01 -07:00
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
value
|
2012-10-08 07:05:01 -07:00
|
|
|
}
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
fn search(&self, hash: uint,
|
2013-03-07 14:38:38 -08:00
|
|
|
op: &fn(x: &Option<Bucket<K, V>>) -> bool) {
|
2012-08-22 21:01:30 -07:00
|
|
|
let _ = self.bucket_sequence(hash, |i| op(&self.buckets[i]));
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-02-26 14:34:00 -05:00
|
|
|
impl<K:Hash + IterBytes + Eq,V>
|
2013-03-14 11:22:51 -07:00
|
|
|
BaseIter<(&'self K, &'self V)> for LinearMap<K, V>
|
2013-02-26 14:34:00 -05:00
|
|
|
{
|
2013-02-07 21:03:13 -05:00
|
|
|
/// Visit all key-value pairs
|
2013-03-14 11:22:51 -07:00
|
|
|
pure fn each(&self, blk: &fn(&(&'self K, &'self V)) -> bool) {
|
2013-02-07 21:03:13 -05:00
|
|
|
for uint::range(0, self.buckets.len()) |i| {
|
|
|
|
let mut broke = false;
|
|
|
|
do self.buckets[i].map |bucket| {
|
|
|
|
if !blk(&(&bucket.key, &bucket.value)) {
|
|
|
|
broke = true; // FIXME(#3064) just write "break;"
|
|
|
|
}
|
|
|
|
};
|
|
|
|
if broke { break; }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
pure fn size_hint(&self) -> Option<uint> { Some(self.len()) }
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<K:Hash + IterBytes + Eq,V> Container for LinearMap<K, V> {
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Return the number of elements in the map
|
2013-03-16 11:11:31 -07:00
|
|
|
pure fn len(&const self) -> uint { self.size }
|
2013-01-23 16:47:27 -05:00
|
|
|
|
|
|
|
/// Return true if the map contains no elements
|
2013-03-16 11:11:31 -07:00
|
|
|
pure fn is_empty(&const self) -> bool { self.len() == 0 }
|
2013-01-21 21:59:19 -05:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<K:Hash + IterBytes + Eq,V> Mutable for LinearMap<K, V> {
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Clear the map, removing all key-value pairs.
|
2013-01-21 17:25:57 -05:00
|
|
|
fn clear(&mut self) {
|
|
|
|
for uint::range(0, self.buckets.len()) |idx| {
|
|
|
|
self.buckets[idx] = None;
|
|
|
|
}
|
|
|
|
self.size = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<K:Hash + IterBytes + Eq,V> Map<K, V> for LinearMap<K, V> {
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Return true if the map contains a value for the specified key
|
2013-01-21 18:22:03 -05:00
|
|
|
pure fn contains_key(&self, k: &K) -> bool {
|
2013-02-10 13:56:03 -08:00
|
|
|
match self.bucket_for_key(k) {
|
2013-01-21 18:22:03 -05:00
|
|
|
FoundEntry(_) => {true}
|
|
|
|
TableFull | FoundHole(_) => {false}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Visit all keys
|
2013-03-07 14:38:38 -08:00
|
|
|
pure fn each_key(&self, blk: &fn(k: &K) -> bool) {
|
2013-02-07 21:03:13 -05:00
|
|
|
self.each(|&(k, _)| blk(k))
|
2013-01-21 18:22:03 -05:00
|
|
|
}
|
|
|
|
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Visit all values
|
2013-03-07 14:38:38 -08:00
|
|
|
pure fn each_value(&self, blk: &fn(v: &V) -> bool) {
|
2013-02-07 21:03:13 -05:00
|
|
|
self.each(|&(_, v)| blk(v))
|
2013-01-21 18:22:03 -05:00
|
|
|
}
|
|
|
|
|
2013-03-13 17:07:23 -04:00
|
|
|
/// Iterate over the map and mutate the contained values
|
|
|
|
fn mutate_values(&mut self, blk: &fn(&'self K,
|
|
|
|
&'self mut V) -> bool) {
|
|
|
|
for uint::range(0, self.buckets.len()) |i| {
|
|
|
|
match self.buckets[i] {
|
|
|
|
Some(Bucket{key: ref key, value: ref mut value, _}) => {
|
|
|
|
if !blk(key, value) { return }
|
|
|
|
}
|
|
|
|
None => ()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Return the value corresponding to the key in the map
|
2013-03-14 11:22:51 -07:00
|
|
|
pure fn find(&self, k: &K) -> Option<&'self V> {
|
2013-02-10 13:56:03 -08:00
|
|
|
match self.bucket_for_key(k) {
|
2013-02-10 15:30:44 -08:00
|
|
|
FoundEntry(idx) => Some(self.value_for_bucket(idx)),
|
|
|
|
TableFull | FoundHole(_) => None,
|
2013-01-23 12:21:58 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Insert a key-value pair into the map. An existing value for a
|
|
|
|
/// key is replaced by the new value. Return true if the key did
|
|
|
|
/// not already exist in the map.
|
2012-10-02 11:37:37 -07:00
|
|
|
fn insert(&mut self, k: K, v: V) -> bool {
|
2012-07-26 16:09:22 -07:00
|
|
|
if self.size >= self.resize_at {
|
|
|
|
// n.b.: We could also do this after searching, so
|
|
|
|
// that we do not resize if this call to insert is
|
|
|
|
// simply going to update a key in place. My sense
|
|
|
|
// though is that it's worse to have to search through
|
|
|
|
// buckets to find the right spot twice than to just
|
|
|
|
// resize in this corner case.
|
|
|
|
self.expand();
|
|
|
|
}
|
|
|
|
|
2012-08-30 19:01:22 -07:00
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
2013-01-24 22:00:58 -05:00
|
|
|
self.insert_internal(hash, k, v)
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Remove a key-value pair from the map. Return true if the key
|
|
|
|
/// was present in the map, otherwise false.
|
2012-08-22 21:01:30 -07:00
|
|
|
fn remove(&mut self, k: &K) -> bool {
|
2013-01-24 22:00:58 -05:00
|
|
|
self.pop(k).is_some()
|
2012-10-08 07:05:01 -07:00
|
|
|
}
|
2013-01-21 18:22:03 -05:00
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub impl<K:Hash + IterBytes + Eq,V> LinearMap<K, V> {
|
2013-01-24 13:36:55 -05:00
|
|
|
/// Create an empty LinearMap
|
2013-01-23 17:06:32 -05:00
|
|
|
static fn new() -> LinearMap<K, V> {
|
|
|
|
linear_map_with_capacity(INITIAL_CAPACITY)
|
|
|
|
}
|
|
|
|
|
2013-02-15 01:35:15 -05:00
|
|
|
/// Reserve space for at least `n` elements in the hash table.
|
|
|
|
fn reserve_at_least(&mut self, n: uint) {
|
|
|
|
if n > self.buckets.len() {
|
|
|
|
let buckets = n * 4 / 3 + 1;
|
|
|
|
self.resize(uint::next_power_of_two(buckets));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-08 07:05:01 -07:00
|
|
|
fn pop(&mut self, k: &K) -> Option<V> {
|
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
|
|
|
self.pop_internal(hash, k)
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
|
2012-10-08 07:05:01 -07:00
|
|
|
fn swap(&mut self, k: K, v: V) -> Option<V> {
|
|
|
|
// this could be faster.
|
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
|
|
|
let old_value = self.pop_internal(hash, &k);
|
|
|
|
|
|
|
|
if self.size >= self.resize_at {
|
|
|
|
// n.b.: We could also do this after searching, so
|
|
|
|
// that we do not resize if this call to insert is
|
|
|
|
// simply going to update a key in place. My sense
|
|
|
|
// though is that it's worse to have to search through
|
|
|
|
// buckets to find the right spot twice than to just
|
|
|
|
// resize in this corner case.
|
|
|
|
self.expand();
|
|
|
|
}
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
self.insert_internal(hash, k, v);
|
2012-10-08 07:05:01 -07:00
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
old_value
|
2012-10-08 07:05:01 -07:00
|
|
|
}
|
|
|
|
|
2013-02-10 15:30:44 -08:00
|
|
|
/// Return the value corresponding to the key in the map, or insert
|
|
|
|
/// and return the value if it doesn't exist.
|
2013-03-14 11:22:51 -07:00
|
|
|
fn find_or_insert(&mut self, k: K, v: V) -> &'self V {
|
2013-02-10 15:30:44 -08:00
|
|
|
if self.size >= self.resize_at {
|
|
|
|
// n.b.: We could also do this after searching, so
|
|
|
|
// that we do not resize if this call to insert is
|
|
|
|
// simply going to update a key in place. My sense
|
|
|
|
// though is that it's worse to have to search through
|
|
|
|
// buckets to find the right spot twice than to just
|
|
|
|
// resize in this corner case.
|
|
|
|
self.expand();
|
|
|
|
}
|
|
|
|
|
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
|
|
|
let idx = match self.bucket_for_key_with_hash(hash, &k) {
|
2013-02-14 13:09:09 -08:00
|
|
|
TableFull => fail!(~"Internal logic error"),
|
2013-02-10 15:30:44 -08:00
|
|
|
FoundEntry(idx) => idx,
|
|
|
|
FoundHole(idx) => {
|
|
|
|
self.buckets[idx] = Some(Bucket{hash: hash, key: k,
|
|
|
|
value: v});
|
|
|
|
self.size += 1;
|
|
|
|
idx
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-02-08 22:21:45 -08:00
|
|
|
unsafe { // FIXME(#4903)---requires flow-sensitive borrow checker
|
|
|
|
::cast::transmute_region(self.value_for_bucket(idx))
|
|
|
|
}
|
2013-02-10 15:30:44 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the value corresponding to the key in the map, or create,
|
|
|
|
/// insert, and return a new value if it doesn't exist.
|
2013-03-14 11:22:51 -07:00
|
|
|
fn find_or_insert_with(&mut self, k: K, f: &fn(&K) -> V) -> &'self V {
|
2013-02-10 15:30:44 -08:00
|
|
|
if self.size >= self.resize_at {
|
|
|
|
// n.b.: We could also do this after searching, so
|
|
|
|
// that we do not resize if this call to insert is
|
|
|
|
// simply going to update a key in place. My sense
|
|
|
|
// though is that it's worse to have to search through
|
|
|
|
// buckets to find the right spot twice than to just
|
|
|
|
// resize in this corner case.
|
|
|
|
self.expand();
|
|
|
|
}
|
|
|
|
|
|
|
|
let hash = k.hash_keyed(self.k0, self.k1) as uint;
|
|
|
|
let idx = match self.bucket_for_key_with_hash(hash, &k) {
|
2013-02-14 13:09:09 -08:00
|
|
|
TableFull => fail!(~"Internal logic error"),
|
2013-02-10 15:30:44 -08:00
|
|
|
FoundEntry(idx) => idx,
|
|
|
|
FoundHole(idx) => {
|
|
|
|
let v = f(&k);
|
|
|
|
self.buckets[idx] = Some(Bucket{hash: hash, key: k,
|
|
|
|
value: v});
|
|
|
|
self.size += 1;
|
|
|
|
idx
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
2013-02-08 22:21:45 -08:00
|
|
|
unsafe { // FIXME(#4903)---requires flow-sensitive borrow checker
|
|
|
|
::cast::transmute_region(self.value_for_bucket(idx))
|
|
|
|
}
|
2013-02-10 15:30:44 -08:00
|
|
|
}
|
|
|
|
|
2013-03-07 14:38:38 -08:00
|
|
|
fn consume(&mut self, f: &fn(K, V)) {
|
2012-10-08 07:05:01 -07:00
|
|
|
let mut buckets = ~[];
|
|
|
|
self.buckets <-> buckets;
|
|
|
|
self.size = 0;
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
do vec::consume(buckets) |_, bucket| {
|
|
|
|
match bucket {
|
|
|
|
None => {},
|
|
|
|
Some(bucket) => {
|
|
|
|
let Bucket{key: key, value: value, _} = bucket;
|
|
|
|
f(key, value)
|
2012-10-08 07:05:01 -07:00
|
|
|
}
|
|
|
|
}
|
2012-07-26 16:09:22 -07:00
|
|
|
}
|
|
|
|
}
|
2012-08-21 15:55:17 -07:00
|
|
|
|
2013-03-14 11:22:51 -07:00
|
|
|
pure fn get(&self, k: &K) -> &'self V {
|
2013-01-23 11:47:43 -05:00
|
|
|
match self.find(k) {
|
2012-09-14 16:02:02 -07:00
|
|
|
Some(v) => v,
|
2013-02-11 19:26:38 -08:00
|
|
|
None => fail!(fmt!("No entry found for key: %?", k)),
|
2012-09-14 16:02:02 -07:00
|
|
|
}
|
|
|
|
}
|
2013-03-04 19:43:14 -08:00
|
|
|
|
|
|
|
/// Return true if the map contains a value for the specified key,
|
|
|
|
/// using equivalence
|
|
|
|
pure fn contains_key_equiv<Q:Hash + IterBytes + Equiv<K>>(
|
|
|
|
&self,
|
|
|
|
key: &Q)
|
|
|
|
-> bool {
|
|
|
|
match self.bucket_for_key_equiv(key) {
|
|
|
|
FoundEntry(_) => {true}
|
|
|
|
TableFull | FoundHole(_) => {false}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return the value corresponding to the key in the map, using
|
|
|
|
/// equivalence
|
|
|
|
pure fn find_equiv<Q:Hash + IterBytes + Equiv<K>>(&self, k: &Q)
|
2013-03-14 11:22:51 -07:00
|
|
|
-> Option<&'self V> {
|
2013-03-04 19:43:14 -08:00
|
|
|
match self.bucket_for_key_equiv(k) {
|
|
|
|
FoundEntry(idx) => Some(self.value_for_bucket(idx)),
|
|
|
|
TableFull | FoundHole(_) => None,
|
|
|
|
}
|
|
|
|
}
|
2012-07-31 17:11:57 -04:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<K:Hash + IterBytes + Eq,V:Eq> Eq for LinearMap<K, V> {
|
2012-12-10 09:00:52 -08:00
|
|
|
pure fn eq(&self, other: &LinearMap<K, V>) -> bool {
|
|
|
|
if self.len() != other.len() { return false; }
|
|
|
|
|
2013-02-07 21:03:13 -05:00
|
|
|
for self.each |&(key, value)| {
|
2013-01-23 11:47:43 -05:00
|
|
|
match other.find(key) {
|
2012-12-10 09:00:52 -08:00
|
|
|
None => return false,
|
|
|
|
Some(v) => if value != v { return false },
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
true
|
2012-12-10 09:00:52 -08:00
|
|
|
}
|
|
|
|
|
2013-01-24 22:00:58 -05:00
|
|
|
pure fn ne(&self, other: &LinearMap<K, V>) -> bool { !self.eq(other) }
|
2012-12-10 09:00:52 -08:00
|
|
|
}
|
2013-01-20 14:18:24 -05:00
|
|
|
|
2013-01-28 10:46:43 -08:00
|
|
|
pub struct LinearSet<T> {
|
2013-01-20 14:18:24 -05:00
|
|
|
priv map: LinearMap<T, ()>
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Hash + IterBytes + Eq> BaseIter<T> for LinearSet<T> {
|
2013-01-20 14:18:24 -05:00
|
|
|
/// Visit all values in order
|
2013-03-07 14:38:38 -08:00
|
|
|
pure fn each(&self, f: &fn(&T) -> bool) { self.map.each_key(f) }
|
2013-01-20 14:18:24 -05:00
|
|
|
pure fn size_hint(&self) -> Option<uint> { Some(self.len()) }
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Hash + IterBytes + Eq> Eq for LinearSet<T> {
|
2013-01-20 17:07:57 -05:00
|
|
|
pure fn eq(&self, other: &LinearSet<T>) -> bool {
|
|
|
|
self.map == other.map
|
|
|
|
}
|
|
|
|
pure fn ne(&self, other: &LinearSet<T>) -> bool {
|
|
|
|
self.map != other.map
|
|
|
|
}
|
2013-01-20 14:18:24 -05:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Hash + IterBytes + Eq> Container for LinearSet<T> {
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Return the number of elements in the set
|
2013-03-16 11:11:31 -07:00
|
|
|
pure fn len(&const self) -> uint { self.map.len() }
|
2013-01-23 16:47:27 -05:00
|
|
|
|
|
|
|
/// Return true if the set contains no elements
|
2013-03-16 11:11:31 -07:00
|
|
|
pure fn is_empty(&const self) -> bool { self.map.is_empty() }
|
2013-01-21 21:59:19 -05:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Hash + IterBytes + Eq> Mutable for LinearSet<T> {
|
2013-01-23 16:47:27 -05:00
|
|
|
/// Clear the set, removing all values.
|
2013-01-21 17:25:57 -05:00
|
|
|
fn clear(&mut self) { self.map.clear() }
|
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
impl<T:Hash + IterBytes + Eq> Set<T> for LinearSet<T> {
|
2013-01-20 14:18:24 -05:00
|
|
|
/// Return true if the set contains a value
|
|
|
|
pure fn contains(&self, value: &T) -> bool {
|
|
|
|
self.map.contains_key(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Add a value to the set. Return true if the value was not already
|
|
|
|
/// present in the set.
|
|
|
|
fn insert(&mut self, value: T) -> bool { self.map.insert(value, ()) }
|
|
|
|
|
|
|
|
/// Remove a value from the set. Return true if the value was
|
|
|
|
/// present in the set.
|
|
|
|
fn remove(&mut self, value: &T) -> bool { self.map.remove(value) }
|
2013-01-29 16:07:11 -05:00
|
|
|
|
2013-01-29 17:04:25 -05:00
|
|
|
/// Return true if the set has no elements in common with `other`.
|
|
|
|
/// This is equivalent to checking for an empty intersection.
|
|
|
|
pure fn is_disjoint(&self, other: &LinearSet<T>) -> bool {
|
|
|
|
iter::all(self, |v| !other.contains(v))
|
|
|
|
}
|
|
|
|
|
2013-01-29 16:07:11 -05:00
|
|
|
/// Return true if the set is a subset of another
|
|
|
|
pure fn is_subset(&self, other: &LinearSet<T>) -> bool {
|
|
|
|
iter::all(self, |v| other.contains(v))
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Return true if the set is a superset of another
|
|
|
|
pure fn is_superset(&self, other: &LinearSet<T>) -> bool {
|
|
|
|
other.is_subset(self)
|
|
|
|
}
|
2013-01-29 19:30:26 -05:00
|
|
|
|
|
|
|
/// Visit the values representing the difference
|
2013-03-07 14:38:38 -08:00
|
|
|
pure fn difference(&self, other: &LinearSet<T>, f: &fn(&T) -> bool) {
|
2013-01-29 19:30:26 -05:00
|
|
|
for self.each |v| {
|
|
|
|
if !other.contains(v) {
|
2013-01-29 21:58:47 -05:00
|
|
|
if !f(v) { return }
|
2013-01-29 19:30:26 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Visit the values representing the symmetric difference
|
|
|
|
pure fn symmetric_difference(&self, other: &LinearSet<T>,
|
2013-03-07 14:38:38 -08:00
|
|
|
f: &fn(&T) -> bool) {
|
2013-01-29 19:30:26 -05:00
|
|
|
self.difference(other, f);
|
|
|
|
other.difference(self, f);
|
|
|
|
}
|
2013-01-29 21:58:47 -05:00
|
|
|
|
|
|
|
/// Visit the values representing the intersection
|
2013-03-07 15:44:21 -08:00
|
|
|
pure fn intersection(&self,
|
|
|
|
other: &LinearSet<T>,
|
|
|
|
f: &fn(&T) -> bool) {
|
2013-01-29 21:58:47 -05:00
|
|
|
for self.each |v| {
|
|
|
|
if other.contains(v) {
|
|
|
|
if !f(v) { return }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Visit the values representing the union
|
2013-03-07 14:38:38 -08:00
|
|
|
pure fn union(&self, other: &LinearSet<T>, f: &fn(&T) -> bool) {
|
2013-01-29 21:58:47 -05:00
|
|
|
for self.each |v| {
|
|
|
|
if !f(v) { return }
|
|
|
|
}
|
|
|
|
|
|
|
|
for other.each |v| {
|
|
|
|
if !self.contains(v) {
|
|
|
|
if !f(v) { return }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2013-01-20 14:18:24 -05:00
|
|
|
}
|
|
|
|
|
2013-02-20 17:07:17 -08:00
|
|
|
pub impl <T:Hash + IterBytes + Eq> LinearSet<T> {
|
2013-01-20 14:18:24 -05:00
|
|
|
/// Create an empty LinearSet
|
2013-01-24 11:47:00 -05:00
|
|
|
static fn new() -> LinearSet<T> { LinearSet{map: LinearMap::new()} }
|
2013-02-15 01:35:15 -05:00
|
|
|
|
|
|
|
/// Reserve space for at least `n` elements in the hash table.
|
|
|
|
fn reserve_at_least(&mut self, n: uint) {
|
|
|
|
self.map.reserve_at_least(n)
|
|
|
|
}
|
2013-01-20 14:18:24 -05:00
|
|
|
}
|
2013-02-10 15:30:44 -08:00
|
|
|
|
2012-10-08 07:05:01 -07:00
|
|
|
#[test]
|
2013-03-02 05:09:36 -05:00
|
|
|
mod test_map {
|
|
|
|
use container::{Container, Mutable, Map, Set};
|
|
|
|
use option::{None, Some};
|
|
|
|
use hashmap::linear::LinearMap;
|
|
|
|
use hashmap::linear;
|
|
|
|
use uint;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_insert() {
|
|
|
|
let mut m = LinearMap::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(m.insert(2, 4));
|
|
|
|
fail_unless!(*m.get(&1) == 2);
|
|
|
|
fail_unless!(*m.get(&2) == 4);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_insert_overwrite() {
|
|
|
|
let mut m = LinearMap::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(*m.get(&1) == 2);
|
|
|
|
fail_unless!(!m.insert(1, 3));
|
|
|
|
fail_unless!(*m.get(&1) == 3);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_insert_conflicts() {
|
|
|
|
let mut m = linear::linear_map_with_capacity(4);
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(m.insert(5, 3));
|
|
|
|
fail_unless!(m.insert(9, 4));
|
|
|
|
fail_unless!(*m.get(&9) == 4);
|
|
|
|
fail_unless!(*m.get(&5) == 3);
|
|
|
|
fail_unless!(*m.get(&1) == 2);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_conflict_remove() {
|
|
|
|
let mut m = linear::linear_map_with_capacity(4);
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(m.insert(5, 3));
|
|
|
|
fail_unless!(m.insert(9, 4));
|
|
|
|
fail_unless!(m.remove(&1));
|
|
|
|
fail_unless!(*m.get(&9) == 4);
|
|
|
|
fail_unless!(*m.get(&5) == 3);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_is_empty() {
|
|
|
|
let mut m = linear::linear_map_with_capacity(4);
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(!m.is_empty());
|
|
|
|
fail_unless!(m.remove(&1));
|
|
|
|
fail_unless!(m.is_empty());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_pop() {
|
|
|
|
let mut m = LinearMap::new();
|
|
|
|
m.insert(1, 2);
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.pop(&1) == Some(2));
|
|
|
|
fail_unless!(m.pop(&1) == None);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_swap() {
|
|
|
|
let mut m = LinearMap::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.swap(1, 2) == None);
|
|
|
|
fail_unless!(m.swap(1, 3) == Some(2));
|
|
|
|
fail_unless!(m.swap(1, 4) == Some(3));
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_find_or_insert() {
|
|
|
|
let mut m = LinearMap::new::<int, int>();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.find_or_insert(1, 2) == &2);
|
|
|
|
fail_unless!(m.find_or_insert(1, 3) == &2);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_find_or_insert_with() {
|
|
|
|
let mut m = LinearMap::new::<int, int>();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.find_or_insert_with(1, |_| 2) == &2);
|
|
|
|
fail_unless!(m.find_or_insert_with(1, |_| 3) == &2);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_consume() {
|
|
|
|
let mut m = LinearMap::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(1, 2));
|
|
|
|
fail_unless!(m.insert(2, 3));
|
2013-03-02 05:09:36 -05:00
|
|
|
let mut m2 = LinearMap::new();
|
|
|
|
do m.consume |k, v| {
|
|
|
|
m2.insert(k, v);
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.len() == 0);
|
|
|
|
fail_unless!(m2.len() == 2);
|
|
|
|
fail_unless!(m2.get(&1) == &2);
|
|
|
|
fail_unless!(m2.get(&2) == &3);
|
2012-07-31 17:11:57 -04:00
|
|
|
}
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
pub fn test_iterate() {
|
|
|
|
let mut m = linear::linear_map_with_capacity(4);
|
|
|
|
for uint::range(0, 32) |i| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.insert(i, i*2));
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
let mut observed = 0;
|
|
|
|
for m.each |&(k, v)| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(*v == *k * 2);
|
2013-03-02 05:09:36 -05:00
|
|
|
observed |= (1 << *k);
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(observed == 0xFFFF_FFFF);
|
2012-07-31 17:11:57 -04:00
|
|
|
}
|
2012-09-07 10:41:07 -07:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
#[test]
|
|
|
|
pub fn test_find() {
|
|
|
|
let mut m = LinearMap::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.find(&1).is_none());
|
2013-03-02 05:09:36 -05:00
|
|
|
m.insert(1, 2);
|
|
|
|
match m.find(&1) {
|
|
|
|
None => fail!(),
|
2013-03-06 13:58:02 -08:00
|
|
|
Some(v) => fail_unless!(*v == 2)
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
2012-09-14 15:19:54 -07:00
|
|
|
}
|
2012-12-10 09:00:52 -08:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
#[test]
|
|
|
|
pub fn test_eq() {
|
|
|
|
let mut m1 = LinearMap::new();
|
|
|
|
m1.insert(1, 2);
|
|
|
|
m1.insert(2, 3);
|
|
|
|
m1.insert(3, 4);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
let mut m2 = LinearMap::new();
|
|
|
|
m2.insert(1, 2);
|
|
|
|
m2.insert(2, 3);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m1 != m2);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
m2.insert(3, 4);
|
2012-12-10 09:00:52 -08:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m1 == m2);
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
2012-12-30 19:35:03 +01:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
#[test]
|
|
|
|
pub fn test_expand() {
|
|
|
|
let mut m = LinearMap::new();
|
2012-12-30 19:35:03 +01:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.len() == 0);
|
|
|
|
fail_unless!(m.is_empty());
|
2012-12-30 19:35:03 +01:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
let mut i = 0u;
|
|
|
|
let old_resize_at = m.resize_at;
|
|
|
|
while old_resize_at == m.resize_at {
|
|
|
|
m.insert(i, i);
|
|
|
|
i += 1;
|
|
|
|
}
|
2012-12-30 19:35:03 +01:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(m.len() == i);
|
|
|
|
fail_unless!(!m.is_empty());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
2012-12-30 19:35:03 +01:00
|
|
|
}
|
2013-01-29 16:07:11 -05:00
|
|
|
|
|
|
|
#[test]
|
2013-03-02 05:09:36 -05:00
|
|
|
mod test_set {
|
|
|
|
use hashmap::linear;
|
|
|
|
use container::{Container, Mutable, Map, Set};
|
|
|
|
use vec;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_disjoint() {
|
|
|
|
let mut xs = linear::LinearSet::new();
|
|
|
|
let mut ys = linear::LinearSet::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(xs.is_disjoint(&ys));
|
|
|
|
fail_unless!(ys.is_disjoint(&xs));
|
|
|
|
fail_unless!(xs.insert(5));
|
|
|
|
fail_unless!(ys.insert(11));
|
|
|
|
fail_unless!(xs.is_disjoint(&ys));
|
|
|
|
fail_unless!(ys.is_disjoint(&xs));
|
|
|
|
fail_unless!(xs.insert(7));
|
|
|
|
fail_unless!(xs.insert(19));
|
|
|
|
fail_unless!(xs.insert(4));
|
|
|
|
fail_unless!(ys.insert(2));
|
|
|
|
fail_unless!(ys.insert(-11));
|
|
|
|
fail_unless!(xs.is_disjoint(&ys));
|
|
|
|
fail_unless!(ys.is_disjoint(&xs));
|
|
|
|
fail_unless!(ys.insert(7));
|
|
|
|
fail_unless!(!xs.is_disjoint(&ys));
|
|
|
|
fail_unless!(!ys.is_disjoint(&xs));
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_subset_and_superset() {
|
|
|
|
let mut a = linear::LinearSet::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.insert(0));
|
|
|
|
fail_unless!(a.insert(5));
|
|
|
|
fail_unless!(a.insert(11));
|
|
|
|
fail_unless!(a.insert(7));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
|
|
let mut b = linear::LinearSet::new();
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(b.insert(0));
|
|
|
|
fail_unless!(b.insert(7));
|
|
|
|
fail_unless!(b.insert(19));
|
|
|
|
fail_unless!(b.insert(250));
|
|
|
|
fail_unless!(b.insert(11));
|
|
|
|
fail_unless!(b.insert(200));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(!a.is_subset(&b));
|
|
|
|
fail_unless!(!a.is_superset(&b));
|
|
|
|
fail_unless!(!b.is_subset(&a));
|
|
|
|
fail_unless!(!b.is_superset(&a));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(b.insert(5));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.is_subset(&b));
|
|
|
|
fail_unless!(!a.is_superset(&b));
|
|
|
|
fail_unless!(!b.is_subset(&a));
|
|
|
|
fail_unless!(b.is_superset(&a));
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_intersection() {
|
|
|
|
let mut a = linear::LinearSet::new();
|
|
|
|
let mut b = linear::LinearSet::new();
|
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.insert(11));
|
|
|
|
fail_unless!(a.insert(1));
|
|
|
|
fail_unless!(a.insert(3));
|
|
|
|
fail_unless!(a.insert(77));
|
|
|
|
fail_unless!(a.insert(103));
|
|
|
|
fail_unless!(a.insert(5));
|
|
|
|
fail_unless!(a.insert(-5));
|
|
|
|
|
|
|
|
fail_unless!(b.insert(2));
|
|
|
|
fail_unless!(b.insert(11));
|
|
|
|
fail_unless!(b.insert(77));
|
|
|
|
fail_unless!(b.insert(-9));
|
|
|
|
fail_unless!(b.insert(-42));
|
|
|
|
fail_unless!(b.insert(5));
|
|
|
|
fail_unless!(b.insert(3));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
|
|
let mut i = 0;
|
|
|
|
let expected = [3, 5, 11, 77];
|
|
|
|
for a.intersection(&b) |x| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(vec::contains(expected, x));
|
2013-03-02 05:09:36 -05:00
|
|
|
i += 1
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(i == expected.len());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
2013-01-29 19:30:26 -05:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
#[test]
|
|
|
|
fn test_difference() {
|
|
|
|
let mut a = linear::LinearSet::new();
|
|
|
|
let mut b = linear::LinearSet::new();
|
2013-01-29 21:58:47 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.insert(1));
|
|
|
|
fail_unless!(a.insert(3));
|
|
|
|
fail_unless!(a.insert(5));
|
|
|
|
fail_unless!(a.insert(9));
|
|
|
|
fail_unless!(a.insert(11));
|
2013-01-29 19:30:26 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(b.insert(3));
|
|
|
|
fail_unless!(b.insert(9));
|
2013-01-29 21:58:47 -05:00
|
|
|
|
2013-03-02 05:09:36 -05:00
|
|
|
let mut i = 0;
|
|
|
|
let expected = [1, 5, 11];
|
|
|
|
for a.difference(&b) |x| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(vec::contains(expected, x));
|
2013-03-02 05:09:36 -05:00
|
|
|
i += 1
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(i == expected.len());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_symmetric_difference() {
|
|
|
|
let mut a = linear::LinearSet::new();
|
|
|
|
let mut b = linear::LinearSet::new();
|
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.insert(1));
|
|
|
|
fail_unless!(a.insert(3));
|
|
|
|
fail_unless!(a.insert(5));
|
|
|
|
fail_unless!(a.insert(9));
|
|
|
|
fail_unless!(a.insert(11));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(b.insert(-2));
|
|
|
|
fail_unless!(b.insert(3));
|
|
|
|
fail_unless!(b.insert(9));
|
|
|
|
fail_unless!(b.insert(14));
|
|
|
|
fail_unless!(b.insert(22));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
|
|
let mut i = 0;
|
|
|
|
let expected = [-2, 1, 5, 11, 14, 22];
|
|
|
|
for a.symmetric_difference(&b) |x| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(vec::contains(expected, x));
|
2013-03-02 05:09:36 -05:00
|
|
|
i += 1
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(i == expected.len());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn test_union() {
|
|
|
|
let mut a = linear::LinearSet::new();
|
|
|
|
let mut b = linear::LinearSet::new();
|
|
|
|
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(a.insert(1));
|
|
|
|
fail_unless!(a.insert(3));
|
|
|
|
fail_unless!(a.insert(5));
|
|
|
|
fail_unless!(a.insert(9));
|
|
|
|
fail_unless!(a.insert(11));
|
|
|
|
fail_unless!(a.insert(16));
|
|
|
|
fail_unless!(a.insert(19));
|
|
|
|
fail_unless!(a.insert(24));
|
|
|
|
|
|
|
|
fail_unless!(b.insert(-2));
|
|
|
|
fail_unless!(b.insert(1));
|
|
|
|
fail_unless!(b.insert(5));
|
|
|
|
fail_unless!(b.insert(9));
|
|
|
|
fail_unless!(b.insert(13));
|
|
|
|
fail_unless!(b.insert(19));
|
2013-03-02 05:09:36 -05:00
|
|
|
|
|
|
|
let mut i = 0;
|
|
|
|
let expected = [-2, 1, 3, 5, 9, 11, 13, 16, 19, 24];
|
|
|
|
for a.union(&b) |x| {
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(vec::contains(expected, x));
|
2013-03-02 05:09:36 -05:00
|
|
|
i += 1
|
|
|
|
}
|
2013-03-06 13:58:02 -08:00
|
|
|
fail_unless!(i == expected.len());
|
2013-03-02 05:09:36 -05:00
|
|
|
}
|
2013-01-29 21:58:47 -05:00
|
|
|
}
|
2013-01-29 16:07:11 -05:00
|
|
|
}
|