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
83
84
85
86
87
88
89
use std::collections::{HashMap, HashSet};
use std::hash::Hash;
pub trait GroupingBy {
type GItem;
fn grouping_by<K, F>(self, key: F) -> HashMap<K, Vec<Self::GItem>>
where
Self: Sized,
F: Fn(&Self::GItem) -> K,
K: Eq + Hash;
fn grouping_by_as_set<K, F>(self, key: F) -> HashMap<K, HashSet<Self::GItem>>
where
Self: Sized,
Self::GItem: Eq + Hash,
F: Fn(&Self::GItem) -> K,
K: Eq + Hash;
}
impl<T: Iterator> GroupingBy for T {
type GItem = <T as Iterator>::Item;
fn grouping_by<K, F>(self, key: F) -> HashMap<K, Vec<Self::GItem>>
where
Self: Sized,
F: Fn(&Self::GItem) -> K,
K: Eq + Hash,
{
let mut map: HashMap<K, Vec<Self::GItem>> = HashMap::new();
for item in self {
map.entry(key(&item)).or_insert_with(Vec::new).push(item);
}
map
}
fn grouping_by_as_set<K, F>(self, key: F) -> HashMap<K, HashSet<Self::GItem>>
where
Self: Sized,
Self::GItem: Eq + Hash,
F: Fn(&Self::GItem) -> K,
K: Eq + Hash,
{
let mut map: HashMap<K, HashSet<Self::GItem>> = HashMap::new();
for item in self {
map.entry(key(&item))
.or_insert_with(HashSet::new)
.insert(item);
}
map
}
}