返回
单循环归并排序:简单高效、占用空间小
前端
2024-01-07 04:21:17
归并排序是一种经典的排序算法,以其稳定性和时间复杂度 O(n log n) 的优点而闻名。然而,传统的多循环归并排序实现方式可能会占用大量空间。单循环/哨兵版归并排序应运而生,它通过简化合并过程,同时保持算法的整体复杂度,解决了这一问题。
单循环/哨兵版归并排序原理
单循环/哨兵版归并排序与传统归并排序遵循相同的 divide-and-conquer 策略,将数组不断拆分为较小的子数组,再将这些子数组合并为一个有序的数组。不同之处在于,它只使用一个循环来完成合并过程,从而降低了空间复杂度。
哨兵节点是一个特殊值,它被添加到每个子数组的末尾,以指示数组的末尾。通过使用哨兵节点,单循环可以遍历合并后的数组,而无需检查每个元素是否为哨兵节点,从而简化了合并过程。
代码实现
以下是用 JavaScript 实现的单循环归并排序算法的代码:
function mergeSort(arr) {
if (arr.length <= 1) {
return arr;
}
const mid = Math.floor(arr.length / 2);
const left = mergeSort(arr.slice(0, mid));
const right = mergeSort(arr.slice(mid));
return merge(left, right);
}
function merge(left, right) {
const merged = [];
let leftIndex = 0;
let rightIndex = 0;
while (leftIndex < left.length || rightIndex < right.length) {
if (leftIndex < left.length && (rightIndex >= right.length || left[leftIndex] <= right[rightIndex])) {
merged.push(left[leftIndex]);
leftIndex++;
} else {
merged.push(right[rightIndex]);
rightIndex++;
}
}
return merged;
}
实例解析
考虑以下数组:
[5, 2, 8, 3, 1, 9, 4, 7, 6]
按照单循环归并排序的步骤,我们可以将数组拆分为:
[5, 2]
[8, 3]
[1, 9]
[4, 7]
[6]
递归地将每个子数组排序,然后将它们合并为:
[2, 5]
[3, 8]
[1, 9]
[4, 7]
最后,将所有子数组合并为有序数组:
[1, 2, 3, 4, 5, 6, 7, 8, 9]
优势与局限
单循环/哨兵版归并排序的主要优势在于:
- 空间复杂度低: 它只使用常数空间,而不管数组的大小。
- 时间复杂度稳定: 与传统归并排序相同,它始终保持 O(n log n) 的时间复杂度,即使在最坏情况下也是如此。
需要注意的是,它在以下情况下可能不如传统多循环归并排序:
- 当数组较大时: 如果数组非常大,额外的空间开销可能变得更加明显。
- 当需要稳定性时: 单循环归并排序不是稳定的排序算法,这意味着具有相同值的元素在排序后的顺序可能与原始数组不同。
总结
单循环/哨兵版归并排序是一种简单高效的排序算法,通过使用哨兵节点和单循环来简化合并过程,同时保持算法的整体复杂度。它特别适用于空间受限的环境,但需要考虑稳定性要求。