Created
March 28, 2016 22:37
-
-
Save jdh30/8e225b7f30715eb588ec to your computer and use it in GitHub Desktop.
Rust implementation of a HashSet benchmark from the F# Journal
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
use std::collections::HashSet; | |
use std::hash::BuildHasherDefault; | |
use std::default::Default; | |
use std::hash::Hasher; | |
pub struct FnvHasher(u64); | |
impl Default for FnvHasher { | |
#[inline] | |
fn default() -> FnvHasher { | |
FnvHasher(0xcbf29ce484222325) | |
} | |
} | |
impl Hasher for FnvHasher { | |
#[inline] | |
fn finish(&self) -> u64 { | |
self.0 | |
} | |
#[inline] | |
fn write(&mut self, bytes: &[u8]) { | |
let FnvHasher(mut hash) = *self; | |
for byte in bytes.iter() { | |
hash = hash ^ (*byte as u64); | |
hash = hash.wrapping_mul(0x100000001b3); | |
} | |
*self = FnvHasher(hash); | |
} | |
} | |
type Set = HashSet<(i32, i32), BuildHasherDefault<FnvHasher>>; | |
fn Empty() -> Set { | |
let fnv = BuildHasherDefault::<FnvHasher>::default(); | |
HashSet::with_hasher(fnv) | |
} | |
fn iterNeighbors<F>(mut f: F, (i, j): (i32, i32)) -> () | |
where F: FnMut((i32, i32)) -> () | |
{ | |
f((i-1, j)); | |
f((i+1, j)); | |
f((i, j-1)); | |
f((i, j+1)); | |
} | |
fn nthLoop(n: i32, s1: Set, s2: Set) -> Set { | |
if n == 0 { | |
return s1; | |
} else { | |
let mut s0 = Empty(); | |
for &p in &s1 { | |
let add = |p| { | |
if !(s1.contains(&p) || s2.contains(&p)) { | |
s0.insert(p); | |
} | |
}; | |
iterNeighbors(add, p); | |
} | |
drop(s2); | |
return nthLoop(n-1, s0, s1); | |
} | |
} | |
fn nth(n: i32, p: (i32, i32)) -> Set { | |
let mut s1 = Empty(); | |
s1.insert(p); | |
let s2 = Empty(); | |
nthLoop(n, s1, s2) | |
} | |
fn main() { | |
let s = nth(2000, (0, 0)); | |
println!("{}", s.len()); | |
} |
@llogiq As far as I can tell it doesn't take the elements of S1 which aren't in S2, it takes the neighbors of the elements in S1 which are in neither S1 nor S2.
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment
Why don't you use
let s0 = s1.difference(&s2).collect();
in Lines 55-61?