美文网首页
sort 排序方法的实现原理

sort 排序方法的实现原理

作者: 浅忆_0810 | 来源:发表于2021-09-18 12:32 被阅读0次

1. sort 方法的基本使用

sort 方法是对数组元素进行排序,默认排序顺序是先将元素转换为字符串,然后再进行排序

arr.sort([compareFunction])

其中 compareFunction 用来指定按某种顺序进行排列的函数,如果省略不写,元素按照转换为字符串的各个字符的 Unicode 位点进行排序

const months = ['March', 'Jan', 'Feb', 'Dec'];
months.sort();
console.log(months); // ["Dec", "Feb", "Jan", "March"]

const array1 = [1, 30, 4, 21, 100000];
array1.sort();
console.log(array1); // [1, 100000, 21, 30, 4]

从上面的执行结果可以看出,如果不加参数,在第二段代码中,21 会排到 4 的前面。这样按照从小到大的逻辑是行不通的,如果想要按照从小到大排序或者从大到小排序,那么上面的代码就需要调整为下面这样

const array1 = [1, 30, 4, 21, 100000];
array1.sort((a,b) => b - a);
console.log(array1); // [100000, 30, 21, 4, 1]

const array1 = [1, 30, 4, 21, 100000];
array1.sort((a,b) => a - b);
console.log(array1); // [1, 4, 21, 30, 100000]

如果指明了 compareFunction 参数 ,那么数组会按照调用该函数的返回值排序,即 a 和 b 是两个将要被比较的元素:

  • 如果 compareFunction(a, b)小于 0,那么 a 会被排列到 b 之前
  • 如果 compareFunction(a, b)等于 0,a 和 b 的相对位置不变
  • 如果 compareFunction(a, b)大于 0,b 会被排列到 a 之前

2. sort 方法的底层实现

下面的源码均来自 V8 源码中关于 sort 排序的摘要,地址:V8 源码 sort 排序部分

如果要排序的元素个数是 n 的时候,那么就会有以下几种情况:

  1. 当 n<=10 时,采用插入排序
  2. 当 n>10 时,采用三路快速排序
  3. 10<n <=1000,采用中位数作为哨兵元素
  4. n>1000,每隔 200~215 个元素挑出一个元素,放到一个新数组中,然后对它排序,找到中间位置的数,以此作为中位数

2.1 为什么元素个数少的时候要采用插入排序

虽然插入排序理论上是平均时间复杂度为 O(n^2) 的算法,快速排序是一个平均 O(nlogn) 级别的算法。但是它们也有最好的时间复杂度情况,而插入排序在最好的情况下时间复杂度是 O(n)

在实际情况中两者的算法复杂度前面都会有一个系数,当 n 足够小的时候,快速排序 nlogn 的优势会越来越小。倘若插入排序的 n 足够小,那么就会超过快排。而事实上正是如此,插入排序经过优化以后,对于小数据集的排序会有非常优越的性能,很多时候甚至会超过快排。因此,对于很小的数据量,应用插入排序是一个非常不错的选择

2.2 为什么要花这么大的力气选择哨兵元素?

因为快速排序的性能瓶颈在于递归的深度,最坏的情况是每次的哨兵都是最小元素或者最大元素,那么进行 partition(一边是小于哨兵的元素,另一边是大于哨兵的元素)时,就会有一边是空的。如果这么排下去,递归的层数就达到了 n , 而每一层的复杂度是 O(n),因此快排这时候会退化成 O(n^2) 级别

因此让哨兵元素尽可能地处于数组的中间位置,让最大或者最小的情况尽可能少

官方实现的 sort 排序算法的代码基本结构

function ArraySort(comparefn) {
  CHECK_OBJECT_COERCIBLE(this, "Array.prototype.sort");
  var array = TO_OBJECT(this);
  var length = TO_LENGTH(array.length);

  return InnerArraySort(array, length, comparefn)
}

function InnerArraySort(array, length, comparefn) {
  // 比较函数未传入
  if (!IS_CALLABLE(comparefn)) {
    comparefn = function (x, y) {
      if (x === y) return 0;
      if (%_IsSmi(x) && %_IsSmi(y)) {
        return SmiLexicographicCompare(x, y);
      }

      x = TO_STRING(x);
      y = TO_STRING(y);

      if (x == y) return 0;
      else return x < y ? -1 : 1;
    };
  }

  function InsertionSort(a, from, to) {
    // 插入排序
    for (var i = from + 1; i < to; i++) {
      var element = a[i];

      for (var j = i - 1; j >= from; j--) {
        var tmp = a[j];
        var order = comparefn(tmp, element);

        if (order > 0) {
          a[j + 1] = tmp;
        } else {
          break;
        }
      }
      a[j + 1] = element;
    }

  }

  // 元素个数大于1000时寻找哨兵元素
  function GetThirdIndex(a, from, to) {   
    var t_array = new InternalArray();
    var increment = 200 + ((to - from) & 15);
    var j = 0;
    from += 1;
    to -= 1;

    for (var i = from; i < to; i += increment) {
      t_array[j] = [i, a[i]];
      j++;
    }

    t_array.sort(function (a, b) {
      return comparefn(a[1], b[1]);
    });
    var third_index = t_array[t_array.length >> 1][0];

    return third_index;
  }

  // 快速排序实现
  function QuickSort(a, from, to) {  
    //哨兵位置
    var third_index = 0;

    while (true) {
      if (to - from <= 10) {
        InsertionSort(a, from, to); // 数据量小,使用插入排序,速度较快
        return;
      }

      third_index = to - from > 1000
        ? GetThirdIndex(a, from, to)
        : from + ((to - from) >> 1); // 小于1000 直接取中点

      // 下面开始快排
      var v0 = a[from];
      var v1 = a[to - 1];
      var v2 = a[third_index];
      var c01 = comparefn(v0, v1);
      if (c01 > 0) {
        var tmp = v0;
        v0 = v1;
        v1 = tmp;
      }

      var c02 = comparefn(v0, v2);
      if (c02 >= 0) {
        var tmp = v0;
        v0 = v2;
        v2 = v1;
        v1 = tmp;
      } else {
        var c12 = comparefn(v1, v2);
        if (c12 > 0) {
          var tmp = v1;
          v1 = v2;
          v2 = tmp;
        }
      }

      a[from] = v0;
      a[to - 1] = v2;
      var pivot = v1;
      var low_end = from + 1;
      var high_start = to - 1;
      a[third_index] = a[low_end];
      a[low_end] = pivot;

      partition: for (var i = low_end + 1; i < high_start; i++) {
        var element = a[i];
        var order = comparefn(element, pivot);
        if (order < 0) {
          a[i] = a[low_end];
          a[low_end] = element;
          low_end++;
        } else if (order > 0) {
          do {
            high_start--;
            if (high_start == i) break partition;
            var top_elem = a[high_start];
            order = comparefn(top_elem, pivot);
          } while (order > 0);

          a[i] = a[high_start];
          a[high_start] = element;
          if (order < 0) {
            element = a[i];
            a[i] = a[low_end];
            a[low_end] = element;
            low_end++;
          }
        }
      }

      // 快排的核心思路,递归调用快速排序方法
      if (to - high_start < low_end - from) {
        QuickSort(a, high_start, to);
        to = low_end;
      } else {
        QuickSort(a, from, low_end);
        from = high_start;
      }
    }
  }
}

3. 总结

排序算法 时间复杂度(最好) 时间复杂度(平均) 时间复杂度(最差) 空间复杂度 稳定性
快速排序 O(nlogn) O(nlogn) O(n^2) O(nlogn) 不稳定
插入排序 O(n) O(n^2) O(n^2) O(1) 稳定

相关文章

  • js排序总结

    用sort()方法实现数字英文排序,用localeCompare()方法实现中文排序。 一、定义 sort():用...

  • 数据结构与算法学习-归并排序和快速排序

    代码准备: 归并排序 归并排序(Merging Sort) 就是利用归并的思想实现排序方法. 它的原理是假设初始序...

  • iOS算法总结-归并排序

    归并排序算法: 归并排序(Merging Sort) 就是利用归并的思想实现的排序方法。它的原理是假设初始序列含有...

  • 数据结构与算法 08: 归并排序

    归并排序算法: 归并排序(Merging Sort) 就是利用归并的思想实现的排序方法。它的原理是假设初始序列含有...

  • pandas高级特性

    1.pandas排序 按标签排序使用sort_index方法实现按标签排序 sort_index方法默认升序排序,...

  • 归并排序

    归并排序( Merging Sort) 就是利用归并的思想实现的排序方法。它的原理是假设初始序列含有n 个记录, ...

  • iOS:归并排序

    归并排序算法 归并排序(Merging Sort) 就是利用归并的思想实现的排序方法。它的原理是假设初始序列含有n...

  • sort 排序方法的实现原理

    1. sort 方法的基本使用 sort 方法是对数组元素进行排序,默认排序顺序是先将元素转换为字符串,然后再进...

  • 排序算法归并排序

    归并排序(Merging Sort)利用归并的思想实现的排序方法。它的原理是假设初始序列含有n个记录,则可以看成是...

  • 数组排序与乱序

    一、规则排序 sort()方法在默认情况下按照升序排列数组项---即从小到大排序,为了实现排序,sort()方法会...

网友评论

      本文标题:sort 排序方法的实现原理

      本文链接:https://www.haomeiwen.com/subject/dnnegltx.html