十大排序算法---JavaScript实现

125 阅读8分钟
排序算法平均时间复杂度最好情况最坏情况空间复杂度排序方式稳定性
冒泡排序O(N2)O(N^2)O(N)O(N)O(N2)O(N^2)O(1)O(1)in-place稳定
选择排序O(N2)O(N^2)O(N2)O(N^2)O(N2)O(N^2)O(1)O(1)in-place不稳定
插入排序O(N2)O(N^2)O(N)O(N)O(N2)O(N^2)O(1)O(1)in-place稳定
希尔排序O(NlogN)O(NlogN)O(Nlog2N)O(Nlog^2N)O(Nlog2N)O(Nlog^2N)O(1)O(1)in-place不稳定
归并排序O(NNlog2N)O(NNlog^2N)O(Nlog2N)O(Nlog^2N)O(Nlog2N)O(Nlog^2N)O(N)O(N)out-place不稳定
快速排序O(Nlog2N)O(Nlog^2N)O(Nlog2N)O(Nlog^2N)O(N2)O(N^2)O(logN)O(logN)in-place不稳定
堆排序O(Nlog2N)O(Nlog^2N)O(Nlog2N)O(Nlog^2N)O(Nlog2N)O(Nlog^2N)O(1)O(1)in-place不稳定
计数排序O(N+K)O(N+K)O(N+K)O(N+K)O(N+K)O(N+K)O(N+K)O(N+K)out-place稳定
桶排序O(N+K)O(N+K)O(N+K)O(N+K)O(N2)O(N^2)O(N+K)O(N+K)out-place稳定
桶排序O(N×K)O(N \times K)O(N×K)O(N \times K)O(N×K)O(N \times K)O(N+K)O(N+K)out-place稳定

1. 冒泡排序

  • 比较相邻的元素。如果第一个比第二个大,就交换他们两个。
  • 对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对。这步做完后,最后的元素会是最大的数。
  • 针对所有的元素重复以上的步骤,除了最后一个。
  • 持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较。
var arr = [1,2,1,7,3,4,9,3,6]
function bubbleSort(arr) {
  for (let i = 0;i < arr.length-1;i++) {
    for (let j = 0;j < arr.length - 1- i;j++){
      if (arr[j] > arr[j+1]) {
        [arr[j], arr[j+1]] = [arr[j+1], arr[j]]
      }
    }
  }
  return arr
}
var r = bubbleSort(arr)
console.log(r);

当输入的数据已经是正序时,算法最快; 当输入的数据是反序时,算法最慢

2. 选择排序

无论什么数据进去都是 O(n²) 的时间复杂度。所以用到它的时候,数据规模越小越好。唯一的好处可能就是不占用额外的内存空间

  • 首先在未排序序列中找到最小(大)元素,存放到排序序列的起始位置。
  • 再从剩余未排序元素中继续寻找最小(大)元素,然后放到已排序序列的末尾。
  • 重复第二步,直到所有元素均排序完毕。
function selectSort(arr) {
  var minIndex = 0
  var len = arr.length
  for (let i=0;i < len-1;i++) {
    minIndex = i
    for (let j=i+1;j < len-1; j++) {
      if (arr[j] < arr[minIndex]) {
        minIndex = j
      }
    }
    [arr[i], arr[minIndex]] = [arr[minIndex], arr[i]]
  }
  return arr
}

3. 插入排序

工作原理是通过构建有序序列,对于未排序数据,在已排序序列中从后向前扫描,找到相应位置并插入。

  • 将待排序序列第一个元素看做一个有序序列,把第二个元素到最后一个元素当成是未排序序列。
  • 从头到尾依次扫描未排序序列,将扫描到的每个元素插入有序序列的适当位置。(如果待插入的元素与有序序列中的某个元素相等,则将待插入元素插入到相等元素的后面。)
function insertSort(arr) {
  var len = arr.length
  var preIndex, curIndex
  for (let i=1;i < len;i++) {
    preIndex = i -1
    current = arr[i]
    while (preIndex >= 0 && arr[preIndex] > curIndex) {
      arr[preIndex+1] = arr[preIndex]
      preIndex--
    }
    arr[preIndex+1] = current
  }
  return arr
}

4. 希尔排序

希尔排序,也称递减增量排序算法,是插入排序的一种更高效的改进版本。 希尔排序是基于插入排序的以下两点性质而提出改进方法的:

  • 插入排序在对几乎已经排好序的数据操作时,效率高,即可以达到线性排序的效率;
  • 但插入排序一般来说是低效的,因为插入排序每次只能将数据移动一位;

希尔排序的基本思想是:先将整个待排序的记录序列分割成为若干子序列分别进行直接插入排序,待整个序列中的记录"基本有序"时,再对全体记录进行依次直接插入排序。

  • 选择一个增量序列 t1t_1t2t_2,……,tkt_k,其中 tit_i > tjt_j, tkt_k = 1;
  • 按增量序列个数 kk,对序列进行 kk 趟排序;
  • 每趟排序,根据对应的增量 tit_i,将待排序列分割成若干长度为 mm 的子序列,分别对各子表进行直接插入排序。仅增量因子为 1 时,整个序列作为一个表来处理,表长度即为整个序列的长度。
function shellSort(arr) {
  var len = arr.length,
    temp,
    gap = 1;
  while (gap < len/3) {
    gap = gap*3 + 1
  }
  for (gap; gap > 0; gap = Math.floor(gap/3)){
    for (var i=gap; i< len;i++) {
      temp = arr[i]
      for (var j = i-gap; j>=0 && arr[j] > temp;j -= gap) {
        arr[j+gap] = arr[j]
      }
      arr[j+gap] = temp
    }
  }
  return arr;
}

5. 归并排序

归并排序(Merge sort)是建立在归并操作上的一种有效的排序算法。该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。

作为一种典型的分而治之思想的算法应用,归并排序的实现由两种方法:

  • 自上而下的递归(所有递归的方法都可以用迭代重写,所以就有了第 2 种方法);
  • 自下而上的迭代; 和选择排序一样,归并排序的性能不受输入数据的影响,但表现比选择排序好的多,因为始终都是 O(nlogn) 的时间复杂度。代价是需要额外的内存空间。
    算法步骤:
  1. 申请空间,使其大小为两个已经排序序列之和,该空间用来存放合并后的序列;
  2. 设定两个指针,最初位置分别为两个已经排序序列的起始位置;
  3. 比较两个指针所指向的元素,选择相对小的元素放入到合并空间,并移动指针到下一位置;
  4. 重复步骤 3 直到某一指针达到序列尾;
  5. 将另一序列剩下的所有元素直接复制到合并序列尾。
  function mergeSort(arr) {
    var len = arr.length
    if (len < 2) {
      return arr
    }
    var middle = Math.floor(len / 2)
    var left = arr.slice(0, middle)
    var right = arr.slice(middle)
    return merge(mergeSort(left), mergeSort(right))
  }
  function merge(left, right) {
    var result = []
    while (left.length && right.length) {
      if (left[0] <= right[0]) {
        result.push(left.shift())
      } else {
        result.push(right.shift())
      }
    }
    while (left.length) {
      result.push(left.shift())
    }
    while(right.length) {
      result.push(right.shift())
    }
    return result
  }

6. 快速排序

是处理大数据最快的排序算法之一,对绝大多数顺序性较弱的随机数列而言,快速排序总是优于归并排序。

  1. 从数列中挑出一个元素,称为 "基准"(pivot);
  2. 重新排序数列,所有元素比基准值小的摆放在基准前面,所有元素比基准值大的摆在基准的后面(相同的数可以到任一边)。在这个分区退出之后,该基准就处于数列的中间位置。这个称为分区(partition)操作;
  3. 递归地(recursive)把小于基准值元素的子数列和大于基准值元素的子数列排序;
  function quickSort(arr, left, right) {
    var len = arr.length,
      partitionIndex,
      left = typeof left != "number" ? 0 : left,
      right = typeof right != "number" ? len - 1 : right;

    if (left < right) {
      partitionIndex = partition(arr, left, right);
      quickSort(arr, left, partitionIndex - 1);
      quickSort(arr, partitionIndex + 1, right);
    }
    return arr;
  }

  function partition(arr, left, right) {
    // 分区操作
    var pivot = left, // 设定基准值(pivot)
      index = pivot + 1;
    for (var i = index; i <= right; i++) {
      if (arr[i] < arr[pivot]) {
        [arr[i], arr[index]] = [arr[index], arr[i]]
        index++
      }
    }
    [arr[pivot], arr[index-1]] = [arr[index-1], arr[pivot]]
    return index-1
  }

7. 堆排序

堆排序(Heapsort)是指利用堆这种数据结构所设计的一种排序算法。堆积是一个近似完全二叉树的结构,并同时满足堆积的性质:即子结点的键值或索引总是小于(或者大于)它的父节点。堆排序可以说是一种利用堆的概念来排序的选择排序。分为两种方法:

  • 大顶堆:每个节点的值都大于或等于其子节点的值,在堆排序算法中用于升序排列;
  • 小顶堆:每个节点的值都小于或等于其子节点的值,在堆排序算法中用于降序排列;
  1. 创建一个堆 H[0……n-1];
  2. 把堆首(最大值)和堆尾互换;
  3. 把堆的尺寸缩小 1,并调用 shift_down(0),目的是把新的数组顶端数据调整到相应位置;
  4. 重复步骤 2,直到堆的尺寸为 1。
var len; // 因为声明的多个函数都需要数据长度,所以把len设置成为全局变量

function buildMaxHeap(arr) {
    // 建立初始大顶堆
    len = arr.length;
    // 对于每个非叶子结点,需要调整其与子节点的大小关系
    for (var i = Math.floor(len / 2); i >= 0; i--) {
      heapify(arr, i);
    }
}

function heapify(arr, i) {
    // 堆调整
    var left = 2 * i + 1,
      right = 2 * i + 2,
      largest = i;

    if (left < len && arr[left] > arr[largest]) {
      largest = left;
    }

    if (right < len && arr[right] > arr[largest]) {
      largest = right;
    }

    if (largest != i) {
      [arr[largest], arr[i]] = [arr[i], arr[largest]]
      // 调整之后的最大值可能会破坏堆结构,所以需要递归调整
      heapify(arr, largest);
    }
}

function heapSort(arr) {
    buildMaxHeap(arr);
    for (var i = arr.length - 1; i > 0; i--) {
      // 将堆首的最大值与最后未排序的数交换
      [arr[0], arr[i]] = [arr[i], arr[0]]
      len--;
      heapify(arr, 0);
    }
    return arr;
}

8. 计数排序

计数排序的核心在于将输入的数据值转化为键存储在额外开辟的数组空间中。作为一种线性时间复杂度的排序,计数排序要求输入的数据必须是有确定范围的整数。

  • 找出待排序的数组中最大和最小的元素
  • 统计数组中每个值为i的元素出现的次数,存入数组C的第i项
  • 对所有的计数累加(从C中的第一个元素开始,每一项和前一项相加)
  • 反向填充目标数组:将每个元素i放在新数组的第C(i)项,每放一个元素就将C(i)减去1
var len; // 因为声明的多个函数都需要数据长度,所以把len设置成为全局变量

function buildMaxHeap(arr) {
  // 建立初始大顶堆
  len = arr.length;
  // 对于每个非叶子结点,需要调整其与子节点的大小关系
  for (var i = Math.floor(len / 2); i >= 0; i--) {
    heapify(arr, i);
  }
}

function heapify(arr, i) {
  // 堆调整
  var left = 2 * i + 1,
      right = 2 * i + 2,
      largest = i;
  
  if (left < len && arr[left] > arr[largest]) {
    largest = left;
  }
  
  if (right < len && arr[right] > arr[largest]) {
    largest = right;
  }
  
  if (largest != i) {
    [arr[largest], arr[i]] = [arr[i], arr[largest]]
    // 调整之后的最大值可能会破坏堆结构,所以需要递归调整
    heapify(arr, largest);
  }
}

function heapSort(arr) {
  buildMaxHeap(arr);
  console.log(arr);
  for (var i = arr.length - 1; i > 0; i--) {
    // 将堆首的最大值与最后未排序的数交换
    [arr[0], arr[i]] = [arr[i], arr[0]]
    len--;
    heapify(arr, 0);
  }
  return arr;
}

9. 桶排序

桶排序是计数排序的升级版。它利用了函数的映射关系,高效与否的关键就在于这个映射函数的确定。为了使桶排序更加高效,我们需要做到这两点:

  1. 在额外空间充足的情况下,尽量增大桶的数量
  2. 使用的映射函数能够将输入的 N 个数据均匀的分配到 K 个桶中

同时,对于桶中元素的排序,选择何种比较排序算法对于性能的影响至关重要。
当输入的数据可以均匀的分配到每一个桶中,算法最快;当输入的数据被分配到了同一个桶中,算法最慢。

function bucketSort(arr, bucketSize) {
  if (arr.length === 0) {
    return arr;
  }
  
  var i;
  var minValue = arr[0];
  var maxValue = arr[0];
  for (i = 1; i < arr.length; i++) {
    if (arr[i] < minValue) {
      minValue = arr[i];                // 输入数据的最小值
    } else if (arr[i] > maxValue) {
      maxValue = arr[i];                // 输入数据的最大值
    }
  }
  
  //桶的初始化
  var DEFAULT_BUCKET_SIZE = 5;            // 设置桶的默认数量为5
  bucketSize = bucketSize || DEFAULT_BUCKET_SIZE;
  var bucketCount = Math.floor((maxValue - minValue) / bucketSize) + 1;  
  var buckets = new Array(bucketCount);
  for (i = 0; i < buckets.length; i++) {
    buckets[i] = [];
  }
  
  //利用映射函数将数据分配到各个桶中
  for (i = 0; i < arr.length; i++) {
    buckets[Math.floor((arr[i] - minValue) / bucketSize)].push(arr[i]);
  }
  
  arr.length = 0;
  for (i = 0; i < buckets.length; i++) {
    insertionSort(buckets[i]);                      // 对每个桶进行排序,这里使用了插入排序
    for (var j = 0; j < buckets[i].length; j++) {
      arr.push(buckets[i][j]);                      
    }
  }
  
  return arr;
}

10. 基数排序

基数排序是一种非比较型整数排序算法,其原理是将整数按位数切割成不同的数字,然后按每个位数分别比较。由于整数也可以表达字符串(比如名字或日期)和特定格式的浮点数,所以基数排序也不是只能使用于整数。

基数排序 vs 计数排序 vs 桶排序

基数排序有两种方法:

这三种排序算法都利用了桶的概念,但对桶的使用方法上有明显差异:

  • 基数排序:根据键值的每位数字来分配桶;
  • 计数排序:每个桶只存储单一键值;
  • 桶排序:每个桶存储一定范围的数值;
//LSD Radix Sort
var counter = [];
function radixSort(arr, maxDigit) {
  var mod = 10;
  var dev = 1;
  for (var i = 0; i < maxDigit; i++, dev *= 10, mod *= 10) {
    for(var j = 0; j < arr.length; j++) {
      var bucket = parseInt((arr[j] % mod) / dev);
      if(counter[bucket]==null) {
        counter[bucket] = [];
      }
      counter[bucket].push(arr[j]);
    }
    var pos = 0;
    for(var j = 0; j < counter.length; j++) {
      var value = null;
      if(counter[j]!=null) {
        while ((value = counter[j].shift()) != null) {
          arr[pos++] = value;
        }
      }
    }
  }
  return arr;
}

参考:www.runoob.com/w3cnote/ten…