-
Notifications
You must be signed in to change notification settings - Fork 5
/
Copy pathquicksort.rs
63 lines (52 loc) · 1.89 KB
/
quicksort.rs
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
#[allow(dead_code)]
fn quicksort<T>(coll: &Vec<T>) -> Vec<T>
where
T: std::cmp::PartialOrd + Sized + Copy + Clone,
{
// To keep return types consistent we preallocate
// the clone for the provided `Vec<T>` in order
// to retrieve it.
//
// This strategy has a performance penalty but
// helps avoid performing algorithms _in situ_
let coll_clone: Vec<T> = coll.clone();
if coll.len() < 2 {
// If theres no elements or theres only
// one element we retrieve the `Vec<T>`
// provided.
// In terms of D&C this is the base case
// which also breaks recursiveness
return coll_clone;
}
// This is the recursive case where we plan
// to minimize the size of the problem thus
// applying a D&C mindset
let pivot = coll_clone.first().unwrap();
let less_than_pivot: Vec<T> = quicksort(&coll.iter().filter(|i| *i < pivot).cloned().collect());
let equal_to_pivot: Vec<T> = coll.iter().filter(|i| *i == pivot).cloned().collect();
let mut greater_than_pivot: Vec<T> =
quicksort(&coll.iter().filter(|i| *i > pivot).cloned().collect());
// Start building the result with the elements less than the pivot
let mut result = less_than_pivot;
// Add the elements equal to the pivot to the result
result.extend(equal_to_pivot);
// Add the elements greater than the pivot to the result
result.append(&mut greater_than_pivot);
result
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn sorts_a_collection() {
let items = vec![8, 2, 4, 6, 5, 7, 10];
let result = quicksort(&items);
assert_eq!(result, vec![2, 4, 5, 6, 7, 8, 10]);
}
#[test]
fn sorts_a_collection_with_repeated_elements() {
let items = vec![8, 2, 4, 6, 5, 7, 10, 8];
let result = quicksort(&items);
assert_eq!(result, vec![2, 4, 5, 6, 7, 8, 8, 10]);
}
}