美文网首页
常见的排序算法

常见的排序算法

作者: lhyt | 来源:发表于2018-02-07 10:48 被阅读0次

    本文是lhyt本人原创,希望用通俗易懂的方法来理解一些细节和难点。转载时请注明出处。文章最早出现于本人github

    0.前言

    相信大家面试的时候都要经历手写什么什么排序这种事情吧,要不就是大概说一下思路。不许用各种语言封装好的函数、api,仅仅用原生的方法把他写出来。虽然看起来没什么意思,但是这也是考察一个人的基础有没有扎实、编程思想好不好的一种方法。

    重要的事情说三遍:

    主要理解快速排序!!!

    主要理解快速排序!!!

    主要理解快速排序!!!

    而且要滚瓜烂熟!

    以下所有的排序都是升序

    1.冒泡排序

    先说最简单的吧,冒泡,就是指数值大的,就是一个大气泡,慢慢冒到后面(水面)去。对于要排序的数组,一个个去遍历,大的数往后移。往后移怎么做到呢,就是遍历的时候,两个数中(第j和j+1个)要是谁比较大,谁在后面(交换位置)。气泡都聚集在水面(大数都从后面聚集在一起,所以每次第二层遍历都会到len-1-j截止)

    时间复杂度O(n^2),如果是已经排好的情况是O(n)

    function bubble(arr){

    for(var i = 0,len = arr.length;i

    for (var j = 0; i < len-1-j; j++) {

    if(arr[j]>arr[j+1]){

    var temp = arr[j+1]

    arr[j] = arr[j+1]

    arr[j] = temp

    }

    }

    }

    return arr

    }

    2.快速排序(重点)

    快排是面试问得最多的了,也是目前用得最多,效率最稳的方法。快速排序的最慢情况是O(n²),升序的时候。但它的数学期望是O(n log n) ,且O(n log n)记号中隐含的常数因子很小,比复杂度稳定等于O(n log n)的归并排序要小很多。对绝大多数顺序性较弱的随机数列而言,快速排序总是占优。

    首先,取一个基准值(我们取第一个,也可以取中间、取最后都行),接着让小于基准值的在左边,大于的在右边,分成左右两堆。然后分别对左边和右边那堆采取同样的操作,取基准值,让基准值左边都是小于他的,右边都是大于他的。一直重复操作,直到全部升序。

    2.1另外开辟两个空数组

    这种方法容易理解,但是依赖另外开辟出来的数组。我们取第一个作为基准,从第二(特别注意,第二个开始!不然栈溢出)个开始遍历,小于基准的数放在左数组(left),大于基准的放在右数组,接着对左数组和右边数组重复操作,再对左边数组的左边和右边进行操作......直到数组长度为1

    function quick(arr){

    if(arr.length <= 1){

    return arr

    }

    var pivot = arr[0]

    var left = []

    var right = []

    var len = arr.length

    for(var i = 1;i

    if(arr[i]>pivot){

    right.push(arr[i])

    }else{

    left.push(arr[i])

    }

    }

    return quick(left).concat([pivot],quick(right))

    }

    当然网上也有人家从中间开始的,中间开始的话,就用splice去除这个值,再给pivot赋值

    2.2直接在原数组操作

    这种比较难理解,但是不会占用其他内存。基准值也是取第一个,quick传入3个参数:数组,子数组的起始位置、子数组的结束位置。第一次赋值肯定是quick(arr),left和right是undefined,left和right默认是第一个和最后一个数的索引。 partitionIndex是基准值索引。

    核心部分是中间判断left

    取第一个基准值,从第二个(index)开始遍历。index是目前最后一个小于基准值的索引(其实是留给基准值自己的一个空位)。如果遍历中第i个有小于基准值的,i与index互换,index再+1,这是什么效果呢,可以看一下:8,5,10,7,6的演示

    基准值是8,从5开始遍历(var i = index; i <= right; i++)。

    i=1发现5<8,arr【1】与arr【1】交换位置,index+1,此时数组不变8,5,10,7,6,但是index已经是2

    i=2发现10>8,跳过,index=2

    i=3发现7<8,arr【3】与arr【2】交换,8,5,7,10,6,index=3

    i=4发现6<8,arr【4】与arr【3】交换,8,5,7,6,10,index=4

    遍历已经完成,把基准值和最后一个小于他的arr【index-1】互换6,5,7,8,10,已经完成了基准值左边都是小于他的、右边都是大于他的目的。

    一个模块的操作完成,基准值索引partitionIndex变成index-1

    接下来就是对分出来的左数组和右数组重复的递归操作

    function quick(arr, left, right) {

    var len = arr.length,

    partitionIndex,

    left = typeof left == 'number' ? left : 0,

    right = typeof right == 'number' ? right : len-1;

    if (left < right) {//保证小于基准值的在左边,大的在右边

    var pivot = left, //基准值是数组的第一个

    index = pivot + 1; //从数组的第二个开始

    for (var i = index; i <= right; i++) {

    if (arr[i] < arr[pivot]) {

    var temp = arr[i];

    arr[i] = arr[index];

    arr[index] = temp;

    index++;

    }

    }

    var temp = arr[pivot];

    arr[pivot] = arr[index - 1];

    arr[index - 1] = temp;

    partitionIndex = index-1;

    quick(arr, left, partitionIndex-1);

    quick(arr, partitionIndex+1, right);

    }

    return arr;

    }

    3.选择排序

    表现最稳定的排序之一,无论什么数据进去都是O(n²)复杂度,但是依赖额外内存保存最小值

    数组长度为len的数组中进行选择排序时:

    取后len个的最小值放数组第一个位置

    取后len-1个的最小值放数组第二个位置

    取后len-2个的最小值放数组第三个位置

    .......

    显然,最后一个就是最大的,前面的已经完成了排序

    function select(arr){

    var min //保存运算过程中最小值索引

    var temp

    for (var i = 0,len = arr.length; i

    min = i //先默认最小是自己

    for(var j = i+1;j

    if(arr[min]>arr[j]){

    min = j //记录更小的值的索引

    }

    }

    temp = arr[i]

    arr[i] = arr[min]

    arr[min] = temp

    }

    return arr

    }

    4.插入排序

    就像打牌一样,自己是怎么整理的呢。拿着一张牌,一个个往下去对比,发现下一个大于(或等于)自己,上一个小于(或等于)自己,那就把他放在中间。复杂度稳定O(n^2),最好的情况是已经排序完成时O(n),依赖额外内存,保存当前遍历的数

    function insert(arr) {

    var len = arr.length;

    var preIndex, current;

    for (var i = 1; i < len; i++) {

    preIndex = i - 1;//小于自己的数的索引

    current = arr[i];

    while(preIndex >= 0 && arr[preIndex] > current) {

    arr[preIndex+1] = arr[preIndex];//如果满足条件,把当前的数插入中间

    preIndex--;

    }

    arr[preIndex+1] = current;

    }

    return arr;

    }

    5.计数排序

    将数组区间取出来(【min,max】),然后对数组进行遍历,计算每一个数组元素在区间【min,max】中出现次数,最后从头到尾把数组写出来。复杂度是O(n),比较稳定,但是依赖额外内存空间

    计数排序需要全部都是整数!

    这里,我们取正整数来试验,所以只要取得最大值就行。arr的值表示bucket的位置,如果bucket的某个位置是undefined,说明arr里面没有这个位置的索引值

    function count(arr) {

    var max = Math.max(...arr)//需要知道最大值的,这里只能作弊了

    var bucket = new Array(max+1),//一个被max+1个undefined填充的数组

    sortedIndex = 0;//重写arr的索引

    for (var i = 0; i < arr.length; i++) {

    if (!bucket[arr[i]]) {//第一次遇到bucket的某个索引值为arr值,将undefined变成0

    bucket[arr[i]] = 0;

    }

    bucket[arr[i]]++;//开始统计

    }

    for (var j = 0; j < max + 1; j++) {

    while(bucket[j] > 0) {

    arr[sortedIndex++] = j;//重写arr

    bucket[j]--;//再一次进入同样的循环,直到没有重复值(0的时候)

    }

    }

    return arr;

    }

    6.基数排序

    先比较个位数的大小,按顺序分类,从头到尾重新取出数字。再进行第二次比较,比较十位数的大小,按顺序分类,再从头到尾取出........直到最高位。要求全是整数。

    我们这里比较两位数以内的

    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]);//和计数排序一样,这里只需要0-9的数组存放

    }

    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;

    }

    7.归并排序

    由名字可以知道,这其实就是一种递归。顺序是比较前面两个=>比较前面两个的后面两个=>比较前面四个=>比较前面四个的后面两个=>比较前面6个的后面两个=>比较前面8个........

    始终都是O(nlogn)的复杂度,不受输入数据影响,但是依赖额外内存

    每一次2个数的小组比较,两个小组的数量一致而且排序方式也是升序,对比起来就是一一对比。大的组比较也是一样。数量不同只有数组长度非2的n次方的尾部的比较。

    function mergeSort(arr) { //采用自上而下的递归方法

    var len = arr.length;

    if(len < 2) {

    return arr;//递归到子数组长度为1为止

    }

    var middle = Math.floor(len / 2),//数组被分成左右两个子数组

    left = arr.slice(0, middle),

    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;

    }

    原文来源于:lhyt的github

    相关文章

      网友评论

          本文标题:常见的排序算法

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