1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
|
use std::{
hash::{BuildHasherDefault, Hasher},
io::Read,
time::Instant,
};
struct MyHasher {
seed: u64,
}
impl Default for MyHasher {
fn default() -> Self {
Self { seed: 0xb5ad4eceda1ce2a9_u64 }
}
}
impl Hasher for MyHasher {
fn write(&mut self, bytes: &[u8]) {
use std::convert::TryInto;
self.seed = u64::from_ne_bytes(bytes.try_into().unwrap()).wrapping_mul(0xc6a4a7935bd1e99d);
}
#[inline]
fn write_u64(&mut self, i: u64) {
self.seed = i.wrapping_mul(0xc6a4a7935bd1e99d);
}
#[inline]
fn finish(&self) -> u64 {
self.seed
}
}
type MyBuildHasher = BuildHasherDefault<MyHasher>;
fn romu_trio(s: &mut [u64]) -> u64 {
let xp = s[0];
let yp = s[1];
let zp = s[2];
s[0] = 15241094284759029579_u64.wrapping_mul(zp);
s[1] = yp.wrapping_sub(xp);
s[1] = s[1].rotate_left(12);
s[2] = zp.wrapping_sub(yp);
s[2] = s[2].rotate_left(44);
return xp;
}
fn main() {
let n = 50_000_000;
let mask = (1 << 25) - 1;
let mut m = std::collections::HashMap::<u64, u64, MyBuildHasher>::default();
m.reserve(n);
let mut rng: [u64; 3] = [1872361123, 123879177, 87739234];
println!("Rust HashMap n = {}, mask = {:#x}", n, mask);
let now = Instant::now();
for _i in 0..n {
let key: u64 = romu_trio(&mut rng) & mask;
*m.entry(key).or_insert(0) += 1;
}
println!("insert : {}ms \tsize : {}", now.elapsed().as_millis(), m.len());
let now = Instant::now();
let mut sum = 0;
for i in 0..mask + 1 { if m.contains_key(&i) { sum += 1; }}
println!("lookup : {}ms \tsum : {}", now.elapsed().as_millis(), sum);
let now = Instant::now();
let mut sum = 0;
for (_, value) in &m { sum += value; }
println!("iterate : {}ms \tsum : {}", now.elapsed().as_millis(), sum);
let mut rng: [u64; 3] = [1872361123, 123879177, 87739234];
let now = Instant::now();
for _ in 0..n {
let key: u64 = romu_trio(&mut rng) & mask;
m.remove(&key);
}
println!("remove : {}ms \tsize : {}", now.elapsed().as_millis(), m.len());
println!("press a key:");
std::io::stdin().bytes().next();
}
|