forked from alexfertel/rust-algorithms
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmerge_sort.rs
62 lines (53 loc) · 1.72 KB
/
merge_sort.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
use crate::sorting::traits::Sorter;
pub fn merge_sort<T: Ord + Copy>(array: &[T]) -> Vec<T> {
if array.len() < 2 {
return array.to_vec();
}
// Get the middle element of the array.
let middle = array.len() / 2;
// Divide the array into left and right halves.
let mut left = merge_sort(&array[..middle]);
let mut right = merge_sort(&array[middle..]);
// Call merge function using parameters as both left array and right array.
merge(&mut left, &mut right)
}
fn merge<T: Ord + Copy>(left: &mut Vec<T>, right: &mut Vec<T>) -> Vec<T> {
let mut result = Vec::new();
for _ in 0..left.len() + right.len() {
if left.is_empty() {
result.append(right);
break;
} else if right.is_empty() {
result.append(left);
break;
} else if left[0] <= right[0] {
result.push(left.remove(0));
} else {
result.push(right.remove(0));
}
}
result
}
// The Merge Sort algorithm is a sorting algorithm that is based on the Divide and Conquer paradigm.
// The Time complexity is `O(nlog(n))` where n is the length of the array.
// Auxillary Space required is `O(n)` Since all the elements are copied to the auxillary space.
pub struct MergeSort;
impl<T> Sorter<T> for MergeSort
where
T: Ord + Copy,
{
fn sort_inplace(array: &mut [T]) {
let result = merge_sort(array);
array.copy_from_slice(&result);
}
fn sort(array: &[T]) -> Vec<T> {
merge_sort(array)
}
}
#[cfg(test)]
mod tests {
use crate::sorting::traits::Sorter;
use crate::sorting::MergeSort;
sorting_tests!(MergeSort::sort, merge_sort);
sorting_tests!(MergeSort::sort_inplace, merge_sort, inplace);
}