排序算法就是研究如何对一个集合进行高效排序的算法,也是在面试时非常常见的面试题型之一。
下面是维基百科堆排序算法的解释:
在计算机科学与数学中,一个排序算法(Sorting algorithm) 是一种能将一串资料依照特定排序方式排列的算法。
在计算机科学所使用的排序算法通常依以下标准分类:
常见的排序算法
冒泡排序可以说是最简单的排序算法了。
冒泡排序的基本思路:
冒泡排序图解:
代码实现:
tsfunction bubbleSort(arr: number[]): number[] {
const n = arr.length;
// 外层循环:0 ~ n-1
for (let i = 0; i < n; i++) {
// 内层循环找到最大值
for (let j = 1; j < n - i; j++) {
if (arr[j - 1] > arr[j]) {
// 交换两个数据
[arr[j - 1], arr[j]] = [arr[j], arr[j - 1]];
}
}
}
return arr;
}
冒泡排序的复杂度分析:
总结:
冒泡排序适用于数据规模较小的情况,因为它的时间复杂度为 O(n2),对于大数据量的排序会变得很慢。
同时,它的实现简单,代码实现也容易理解,适用于学习排序算法的初学者
但是,在实际的应用中,冒泡排序并不常用,因为它的效率很低
因此,在实际应用中,冒泡排序通常被更高效的排序算法代替,如快速排序、归并排序等
选择排序也是一种简单的排序算法。
它的基本思想:
选择排序的主要优点与数据移动有关
n
个元素的表进行排序总共进行 至多 n-1
次交换。代码实现:
tsfunction selectionSort(arr: number[]): number[] {
const n = arr.length;
// 外层循环作用:经过多少轮的找最小值
for (let i = 0; i < n; i++) {
let minIndex = i;
// 内存循环作用:每次找到最小值
for (let j = i + 1; j < n; j++) {
if (arr[j] < arr[minIndex]) {
minIndex = j;
}
}
if (minIndex !== i) {
[arr[minIndex], arr[i]] = [arr[i], arr[minIndex]];
}
}
return arr;
}
复杂度分析:
选择排序的总结:
插入排序就像我们打扑克牌时,摸到一张新牌需要插入到手牌中的合适位置一样。
而插入排序的实现方式就是与打牌类似的。
插入排序的图解:
、
代码实现:
tsfunction insertionSort(arr: number[]): number[] {
const n = arr.length;
for (let i = 1; i < n; i++) {
// 内层循环
const newNum = arr[i];
let j = i - 1;
while (arr[j] > newNum && j >= 0) {
arr[j + 1] = arr[j];
j--;
}
arr[j + 1] = newNum;
}
return arr;
}
复杂度分析:
总结:
归并排序是一种典型的分而治之思想的算法,它的算法复杂度为 O(nlogn)
,是一种比较高效的排序算法。
它的基本思想:
merge
)成一个整体有序的数组归并排序的图解:
代码实现:
tsconst mergeSort = function (arr: number[]) {
if (arr.length === 1) return arr;
// 1. 分解(divide):对数组进行分解(分解成链各个小数组)
// 1.1 递归地切割数组得到左边的数组left 和 右边的数组right
const mid = arr.length >> 1;
const left: number[] = mergeSort(arr.slice(0, mid));
const right: number[] = mergeSort(arr.slice(mid));
// 2. 合并(merge):将两个子数组进行合并(双指针)
// 2.1 定义双指针
const result: number[] = [];
let leftIndex = 0;
let rightIndex = 0;
while (leftIndex < left.length && rightIndex < right.length) {
if (left[leftIndex] < right[rightIndex]) {
result[leftIndex + rightIndex] = left[leftIndex++];
} else {
result[leftIndex + rightIndex] = right[rightIndex++];
}
}
// 2.2 判断是否一个数组中海油剩余元素
// 循环完左边还有剩余
while (leftIndex < left.length) {
result[leftIndex + rightIndex] = left[leftIndex++];
}
// 循环完右边还有剩余
while (rightIndex < right.length) {
result[leftIndex + rightIndex] = right[rightIndex++];
}
return result;
};
复杂度分析:
总结:
快速排序是一种经典基于分治思想的排序算法
它的基本思想:
快速排序是一种原地排序算法,不需要额外的数组空间,同时它的时间复杂度为O(nlogn)。
快速排序图解:
代码实现:
tsconst quickSort = function (arr: number[]): number[] {
partition(0, arr.length - 1);
function partition(left: number, right: number) {
if (left >= right) return;
// 1. 找到基准元素(pivot轴心)
const pivot = arr[right];
// 2. 双指针进行交换操作(左边都是比 pivot 小的数字,右边都是比 pivot 大的数字)
let i: number = left;
let j: number = right - 1;
while (i <= j) {
// 找到一个比 pivot 大的元素
while (arr[i] < pivot) {
i++;
}
// 找到一个比 pivot 小的元素
while (arr[j] > pivot) {
j--;
}
// 说明我们已经找到了 比pivot大的元素arr[i] 和 比pivot小的元素arr[j]
if (i <= j) {
[arr[i], arr[j]] = [arr[j], arr[i]];
i++;
j--;
}
console.log({ arr });
}
// 将 pivot 放在正确的位置
[arr[i], arr[right]] = [arr[right], arr[i]];
// 左右继续划分区域(partition)
partition(left, j);
partition(i + 1, right);
}
return arr;
};
总结:
堆排序是一种基于比较的排序算法,它的核心思想是使用二叉堆来维护一个有序序列
1
堆排序的图解:
代码实现:
tsfunction heapSort(arr: number[]): number[] {
// 1. 获取数组的长度
const n = arr.length;
// 2. 对 arr 进行原地建堆
// 2.1 从第一个非叶子节点开始进行下滤操作
const start = Math.floor(n / 2 - 1);
for (let i = start; i >= 0; i--) {
// 2.2 进行下滤操作
heapifyDown(arr, n, i);
}
// 3. 对最大堆进行排序
for (let i = n - 1; i > 0; i--) {
[arr[0], arr[i]] = [arr[i], arr[0]];
heapifyDown(arr, i, 0);
}
return arr;
}
function heapifyDown(arr: number[], n: number, index: number) {
while (2 * index + 1 < n) {
// 1. 获取左右子节点的索引
const leftChildIndex = 2 * index + 1;
const rightChildIndex = 2 * index + 2;
// 2. 找出左右子节点较大的值
let largerIndex = leftChildIndex;
if (rightChildIndex < n && arr[rightChildIndex] > arr[leftChildIndex]) {
largerIndex = rightChildIndex;
}
// 3. 判断 index 位置的值比更大的子节点,直接 break
if (arr[index] >= arr[largerIndex]) {
break;
}
// 4. 和更大位置的进行交换
[arr[index], arr[largerIndex]] = [arr[largerIndex], arr[index]];
index = largerIndex;
}
}
时间复杂度:O(nlogn)
n/2
次堆的向下调整操作,因此它的时间复杂度为 O(n)
。n
次堆的删除的最大值操作,每次操作都需要将堆的最后一个元素与堆顶元素交换,然后向下调整堆。O(logn)
,因此整个排序过程的时间复杂度为 O(nlogn)
空间复杂度:O(1)
堆排序的总结:
O(logn)
的优秀性能,并且由于它只使用了常数个辅助变量来存储堆的信息,因此空间复杂度为 O(1)
。本文作者:叶继伟
本文链接:
版权声明:本博客所有文章除特别声明外,均采用 BY-NC-SA 许可协议。转载请注明出处!