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
use std::collections::{BTreeMap, HashSet};
use std::hash::Hash;
use std::ops::Deref;
#[derive(Debug, Default)]
pub struct MultiMap<K, V>
where
K: Ord + Eq + Hash,
V: Eq + Hash,
{
backing: BTreeMap<K, HashSet<V>>,
}
impl<K, V> MultiMap<K, V>
where
K: Ord + Eq + Hash,
V: Eq + Hash,
{
pub fn insert(&mut self, key: K, value: V) -> bool {
self.backing
.entry(key)
.or_insert_with(Default::default)
.insert(value)
}
pub fn remove(&mut self, key: &K, value: &V) -> bool {
if let Some(values) = self.backing.get_mut(key) {
let only_one_left = values.len() == 1;
if !only_one_left {
return values.remove(value);
}
if value
!= values
.iter()
.next()
.expect("We know there is only one element in collection, tested above; qed")
{
return false;
}
} else {
return false;
}
self.backing.remove(key).is_some()
}
}
impl<K, V> Deref for MultiMap<K, V>
where
K: Ord + Eq + Hash,
V: Eq + Hash,
{
type Target = BTreeMap<K, HashSet<V>>;
fn deref(&self) -> &Self::Target {
&self.backing
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn insert_and_get() {
let mut map: MultiMap<u8, u32> = MultiMap::default();
map.insert(1u8, 3u32);
map.insert(2u8, 6u32);
map.insert(1u8, 9u32);
let set: HashSet<u32> = [3u32, 9u32].iter().cloned().collect();
assert_eq!(map.get(&1u8).unwrap(), &set);
}
}