面试过程中常见的排序算法问题你见个?附常见排序算法源代码(面试必问:常见排序算法问题汇总及源代码解析)

原创
ithorizon 6个月前 (10-21) 阅读数 22 #后端开发

在软件开发面试中,排序算法是一个频繁被问到的话题。掌握常见的排序算法及其实现,对于面试成就至关重要。本文将汇总面试中常见的排序算法问题,并提供相应的源代码解析。

一、冒泡排序

冒泡排序是一种明了的排序算法,它重复地遍历待排序的数列,一次比较两个元素,如果它们的顺序失误就把它们交换过来。遍历数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。

function bubbleSort(arr) {

let n = arr.length;

let swapped;

do {

swapped = false;

for (let i = 1; i < n; i++) {

if (arr[i - 1] > arr[i]) {

let temp = arr[i - 1];

arr[i - 1] = arr[i];

arr[i] = temp;

swapped = true;

}

}

} while (swapped);

return arr;

}

二、选择排序

选择排序是一种明了直观的排序算法。它的工作原理是:第一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,然后再从剩余的元素中寻找最小(大)元素,然后放到已排序的序列的末尾。

function selectionSort(arr) {

let n = arr.length;

for (let i = 0; i < n - 1; i++) {

let minIndex = i;

for (let j = i + 1; j < n; j++) {

if (arr[j] < arr[minIndex]) {

minIndex = j;

}

}

let temp = arr[minIndex];

arr[minIndex] = arr[i];

arr[i] = temp;

}

return arr;

}

三、插入排序

插入排序是一种明了直观的排序算法。它的工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。插入排序在实现上,通常采用in-place排序(即只需用到O(1)的额外空间的排序)。

function insertionSort(arr) {

let n = arr.length;

for (let i = 1; i < n; i++) {

let key = arr[i];

let j = i - 1;

while (j >= 0 && arr[j] > key) {

arr[j + 1] = arr[j];

j = j - 1;

}

arr[j + 1] = key;

}

return arr;

}

四、飞速排序

飞速排序是一种分而治之的策略,它将原始数组分为较小的数组(但它们没有排序),然后再递归地将这些小数组排序,以实现整体排序。

function quickSort(arr, left, right) {

if (left < right) {

let pivotIndex = partition(arr, left, right);

quickSort(arr, left, pivotIndex - 1);

quickSort(arr, pivotIndex + 1, right);

}

return arr;

}

function partition(arr, left, right) {

let pivot = arr[right];

let i = left - 1;

for (let j = left; j < right; j++) {

if (arr[j] < pivot) {

i++;

[arr[i], arr[j]] = [arr[j], arr[i]];

}

}

[arr[i + 1], arr[right]] = [arr[right], arr[i + 1]];

return i + 1;

}

五、归并排序

归并排序是采用了分治法的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。

function mergeSort(arr) {

if (arr.length < 2) {

return arr;

}

const middle = Math.floor(arr.length / 2);

const left = arr.slice(0, middle);

const right = arr.slice(middle);

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

}

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 heapSort(arr) {

let n = arr.length;

// Build heap (rearrange array)

for (let i = Math.floor(n / 2) - 1; i >= 0; i++) {

heapify(arr, n, i);

}

// One by one extract an element from heap

for (let i = n - 1; i > 0; i--) {

// Move current root to end

[arr[0], arr[i]] = [arr[i], arr[0]];

// call max heapify on the reduced heap

heapify(arr, i, 0);

}

return arr;

}

function heapify(arr, n, i) {

let largest = i; // Initialize largest as root

let l = 2 * i + 1; // left = 2*i + 1

let r = 2 * i + 2; // right = 2*i + 2

// If left child is larger than root

if (l < n && arr[l] > arr[largest]) {

largest = l;

}

// If right child is larger than largest so far

if (r < n && arr[r] > arr[largest]) {

largest = r;

}

// If largest is not root

if (largest !== i) {

[arr[i], arr[largest]] = [arr[largest], arr[i]];

// Recursively heapify the affected sub-tree

heapify(arr, n, largest);

}

}

七、希尔排序

希尔排序是插入排序的一种更高效的改进版本。希尔排序是非稳定排序算法。希尔排序使用嵌套循环对数组进行遍历,凭借步长将数组分组,然后对每组使用插入排序。

function shellSort(arr) {

let n = arr.length;

for (let gap = Math.floor(n / 2); gap > 0; gap = Math.floor(gap / 2)) {

for (let i = gap; i < n; i += 1) {

let temp = arr[i];

let j;

for (j = i; j >= gap && arr[j - gap] > temp; j -= gap) {

arr[j] = arr[j - gap];

}

arr[j] = temp;

}

}

return arr;

}

八、总结

以上是面试中常见的几种排序算法及其源代码实现。领会这些算法的原理和实现细节,对于应对技术面试中的排序算法问题至关重要。在实际应用中,选择合适的排序算法可以减成本时间程序的高效,优化用户体验。

面试时,除了算法实现,面试官也许还会问到算法的时间复杂化度和空间复杂化度,以及各种排序算法的适用场景。于是,在准备面试时,不仅要掌握算法的实现,还要领会其理论背景和应用场景。

期待本文能够帮助你在面试中更好地应对排序算法相关的问题。


本文由IT视界版权所有,禁止未经同意的情况下转发

文章标签: 后端开发


热门