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
use std::mem;
pub fn dedup_by<T, F>(v: &mut Vec<T>, mut f: F)
where F: FnMut(&T, &T) -> bool
{
unsafe {
let ln = v.len();
if ln <= 1 {
return;
}
let p = v.as_mut_ptr();
let mut r: usize = 1;
let mut w: usize = 1;
while r < ln {
let p_r = p.offset(r as isize);
let p_wm1 = p.offset((w - 1) as isize);
if !f(&*p_r, &*p_wm1) {
if r != w {
let p_w = p_wm1.offset(1);
mem::swap(&mut *p_r, &mut *p_w);
}
w += 1;
}
r += 1;
}
v.truncate(w);
}
}
#[cfg(test)]
mod tests {
use super::dedup_by;
#[test]
fn test_dedup_by() {
let mut subs = vec!["a", "b", "c", "c", "b"];
let d = vec!["a", "b", "c"];
subs.sort();
let mut subs_clone = subs.clone();
dedup_by(&mut subs, |a, b| a == b);
subs_clone.dedup();
assert_eq!(subs, d);
assert_eq!(subs, subs_clone);
}
}