-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmerge-sort.js
73 lines (67 loc) · 1.78 KB
/
merge-sort.js
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
// tag::merge[]
/**
* Merge two arrays in ascending order
*
* @param {array} array1 sorted array 1
* @param {array} array2 sorted array 2
* @returns {array} merged arrays in asc order
*
* @example
* merge([2,5,9], [1,6,7]) => [1, 2, 5, 6, 7, 9]
*/
function merge(array1, array2 = []) {
const mergedLength = array1.length + array2.length;
const mergedArray = Array(mergedLength);
// merge elements on a and b in asc order. Run-time O(a + b)
for (let index = 0, i1 = 0, i2 = 0;
index < mergedLength; index++) { // <1>
if (i2 >= array2.length
|| (i1 < array1.length && array1[i1] <= array2[i2])) {
mergedArray[index] = array1[i1]; // <2>
i1 += 1;
} else {
mergedArray[index] = array2[i2]; // <2>
i2 += 1;
}
}
return mergedArray; // <3>
}
// end::merge[]
// tag::splitSort[]
/**
* Split array in half recursively until two or less elements are left.
* Sort these two elements and combine them back using the merge function.
* @param {Array} array
* @example
* splitSort([3, 2, 1]) => [1, 2, 3]
* splitSort([3]) => [3]
* splitSort([3, 2]) => [2, 3]
*/
function splitSort(array) {
const size = array.length;
// base case
if (size < 2) {
return array;
} if (size === 2) {
return array[0] < array[1] ? array : [array[1], array[0]]; // <1>
}
// recursive split in half and merge back
const half = Math.ceil(size / 2);
return merge( // <3>
splitSort(array.slice(0, half)), // <2>
splitSort(array.slice(half)), // <2>
);
}
// end::splitSort[]
// tag::sort[]
/**
* Merge sort
* Runtime: O(n log n)
* @param {Array|Set} collection elements to be sorted
*/
function mergeSort(collection) {
const array = Array.from(collection); // <1>
return splitSort(array);
}
// end::sort[]
module.exports = mergeSort;