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
let arr = [49, 38, 65, 97, 76, 13, 27, 49];
console.log(mergeSort(arr))

function merge(left, right) {
let tmp = [];

while (left.length && right.length) {
if (left[0] < right[0])
tmp.push(left.shift());
else
tmp.push(right.shift());
}

return tmp.concat(left, right);
}

function mergeSort(a) {
if (a.length === 1)
return a;

let mid = Math.floor(a.length / 2);
let left = a.slice(0, mid);
let right = a.slice(mid);

return merge(mergeSort(left), mergeSort(right));
}

这段合并排序的代码相当简单直观,但是mergeSort()函数会导致很频繁的自调用。一个长度为n的数组最终会调用mergeSort() 2*n-1次,这意味着如果需要排序的数组长度很大会在某些栈小的浏览器上发生栈溢出错误。

来源自网络

迭代

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
console.log(mergeSort([1, 3, 4, 2, 5, 0, 8, 10, 4]));

function merge(left, right) {
let result = [];

while (left.length && right.length) {
if (left[0] < right[0])
result.push(left.shift());
else
result.push(right.shift());
}

return result.concat(left, right);
}

function mergeSort(a) {
if (a.length === 1)
return a;

let work = [];
for (let i = 0, len = a.length; i < len; i++)
work.push([a[i]]);

work.push([]); // 如果数组长度为奇数,避免下面work[k+1]越界

for (let lim = len; lim > 1; lim = Math.floor((lim + 1) / 2)) {
for (let j = 0, k = 0; k < lim; j++, k += 2)
work[j] = merge(work[k], work[k + 1]);

work[j] = []; //见下面注释
}

return work[0];
}

迭代时为什么要work[j] = []?

因为merge时比较大的值并未从数组中删去,虽然会被覆盖掉,但每趟合并的最后一个work[k+1]里的值如果有剩余,就会被保留下来,如果数组长度为奇数,则在下一趟合并时就会和最后一组有序子序列一起合并,导致出现多余的数字的错误。

如上图,1和3已经合并,但由于1比3小,所以执行了merge函数里result.push(left.shift()),而right其实是没有进行操作的,所以work[1]里的3还存在。

这里合并完后,分为了3组有序子序列[1,2,3,4][0,5,8,10][4],但由于遗留下来的work[3]里的[8,10],下一趟合并时会和索引为2的有序子序列[4],进行合并。

从而导致排序后多了个8和10。所以在每趟合并后加入work[j] = [],将最后一组有序子序列索引+1的数据清空。

作者

DullSword

发布于

2018-07-27

更新于

2024-11-05

许可协议

评论