js排序算法

一、 冒泡排序

冒泡排序(Bubble Sort),是一种计算机科学领域的较简单的排序算法。它重复地走访过要排序的数列,一次比较两个元素,如果他们的顺序错误就把他们交换过来。走访数列的工作是重复地进行直到没有再需要交换,也就是说该数列已经排序完成。这个算法的名字由来是因为越大的元素会经由交换慢慢“浮”到数列的顶端。

平均复杂度:o(n^2) 空间复杂度:o(1) 稳定性:稳定

步骤:
1、比较相邻的元素。如果第一个比第二个大,就交换他们两个;
2、对每一对相邻元素作同样的工作,从开始第一对到结尾的最后一对,这样,最后的元素会是最大的数;
3、针对所有的元素重复以上的步骤,除了最后一个;
4、持续每次对越来越少的元素重复上面的步骤,直到没有任何一对数字需要比较。

代码:

    // 冒泡排序 每次将最大的数值冒泡到最右面,并且执行下次外部循环时将右部最大数剔除,
    // 外部循环执行次数为: 数组长度-1(即假设最坏情况每个数都需要位移需要位移数组长度-1个数)
    // 内部循环次数为: 数组长度 - 当前已经选出右部最大数个数(i) - 1 (位移数比数组长度少一)
let a = [1, 3, 6, 2, 7];

for (let i = 0; i < a.length - 1; i++) {
        for (let j = 0; j < a.length - i - 1; j++) {
            if (a[j] > a[j + 1]) {
                let c = a[j];
                a[j] = a[j + 1];
                a[j + 1] = c;
                // [a[j], a[j + 1]] = [a[j + 1], a[j]]
            }
        }
    }

for (let i = a.length; i > 0; i--) {
        for (let j = 0; j < i - 1; j++) {
            if (a[j] > a[j + 1]) {
                [a[j], a[j + 1]] = [a[j + 1], a[j]]
            }
        }
    }

二、选择排序

选择排序(Selection sort)是一种简单直观的排序算法。它的工作原理是每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完。 选择排序是不稳定的排序方法。

每一次从待排序的数据元素中选出最小(或最大)的一个元素,存放在序列的起始位置,直到全部待排序的数据元素排完。
平均复杂度:o(n^2) 空间复杂度:o(1) 稳定性:不稳定

步骤:

1、每一次循环,找到最小的那个数,并用变量记住它的索引
2、然后将最小值放在它该在的位置上
3、持续对越来越少的元素重复上面的步骤

代码:

// 第一次循环找到数组中最小的数,并将其放在左面
// 之后的循环在数组剩余数中找到第二小的以此类推
for (let i = 0; i < a.length - 1; i++) {
        for (let j = i + 1; j < a.length; j++) {
            if (a[i] > a[j]) {
                let c = a[j];
                a[j] = a[i];
                a[i] = c;
                // [a[j], a[i]] = [a[i], a[j]]
            }
        }
    }

待确认(两种循环执行次数相同,但比选择排序交换次数要多)

// 其实就是先选取前两个数对比,大数放后面。第二次加入第三个数逐项对比,如果第三个数大于第一个数就和第一个数交换,之后第二个数和交换后的第三个数对比交换。即一个逐渐增加的数组进行排序,2个数据排序,3个数据排序,新加入的数据每个对比,符合条件交换(这步比插入排序多了,因为将已经排好的数组又便利一遍排序)
// 首先外部循环确定当前要比对的值a[i]
// 然后内部循环逐项对比数组a[i]之前的数a[j]与a[i]做比对,前一项大于后一项时交换位置。
for (let i = 1; i < a.length; i++) {
        for (let j = 0; j < i; j++) {
            if (a[i] < a[j]) {
                [a[j], a[i]] = [a[i], a[j]]
            }
        }
    }

三、插入排序

插入排序基本操作就是将一个数据插入到已经排好序的有序数据中,从而得到一个新的、个数加一的有序数据,算法适用于少量数据的排序,时间复杂度为O(n^2)。是稳定的排序方法。插入排序的基本思想是:每步将一个待排序的纪录,按其关键码值的大小插入前面已经排序的文件中适当位置上,直到全部插入完为止。

// 先确定第一个基准数据,由于需要对比所以第一个基准数据为a[1], i为1。 (这里基准数据为currentValue)
// 拿到基准数据后再确定基准数据的前一项(这里为a[flag])进行对比如果前一项大于基准数据则前一项向后移一位(a[flag + 1] = a[flag])
// 移动完毕因为前面还有其他更多的数(前一项的前一项的前一项...)都需要和基准数比如果比基准数大则向后移一位
// 直到基准数大于(该前一项)将基准数插入到(该前一项的后面)( a[flag + 1] = currentValue;)
// while循环停止的判断标准是执行到基准数大于等于(前一项)(a[flag] <= currentValue)
// 或者(前一项)不存在即数组的-1项时跳出循环 (flag < 0)
// 由于数组的-1项为undefined undefined和数字对比一定为false所以一定跳出while循环所以flag >= 0判断条件可以不写
for (let i = 1; i < a.length; i++) {
        flag = i - 1;
        currentValue = a[i];
        // flag >= 0  可以不写因为a[flag]为undefined undefined和任何数对比都为false可以终止while循环
        while (flag >= 0 && a[flag] > currentValue) {
            a[flag + 1] = a[flag]
            flag--;
        }
        a[flag + 1] = currentValue;
    }

四、快速排序

快速排序(Quicksort)是对冒泡排序的一种改进。它的基本思想是:通过一趟排序将要排序的数据分割成独立的两部分,其中一部分的所有数据都比另外一部分的所有数据都要小,然后再按此方法对这两部分数据分别进行快速排序,整个排序过程可以递归进行,以此达到整个数据变成有序序列。

// 先选取一个基准数据(例如a[0]),定义为flag用来和这个基准数据做对比,并且从数组中删除该基准数。
// 然后定义空数组left和right, 将数组每项和基准数做对比,如果小于基准数放在left数组中,大于的话放在right数组中
// 分别递归调用处理left和right数组,将处理好的left数组、中间选取的基准数据和right数组合并
// 因为每次递归调用都要删除一个基准数据所以到最后这个数组长度为1时就直接返回数组
var a = [2, 8, 7, 5, 1, 9]

    function quickSort(arr) {
        if (arr.length <= 1) {
            return arr
        }
        let flag = arr[0];
        let left = [];
        let right = [];
        arr.splice(0, 1);
        for (let i = 0; i < arr.length; i++) {
            if (arr[i] < flag) {
                left.push(arr[i])
            } else {
                right.push(arr[i])
            }
        }
        return quickSort(left).concat(flag, quickSort(right))
    }

    console.log(quickSort(a))

五、归并排序

归并排序是建立在归并操作上的一种有效的排序算法,该算法是采用分治法(Divide and Conquer)的一个非常典型的应用。将已有序的子序列合并,得到完全有序的序列;即先使每个子序列有序,再使子序列段间有序。若将两个有序表合并成一个有序表,称为二路归并。
(1) 把长度为n的输入序列分成两个长度为n/2的子序列;
(2)对这两个子序列分别采用归并排序;
(3) 将两个排序好的子序列合并成一个最终的排序序列。

download.png

// 将数组从中间分开为两个数组left和right,递归调用将分开的两个数组再次分别分为两个数组,直到该数组长度为1时
// 递归调用的同时将left和right第一项做对比小的放入到空数组中并删除最小项,并且将剩余的数组left和right数组合并(因为不能确定到底最后剩余的数组中left和right哪个大),其实每次对比的时候左右数组都是已经排序好的数组所以只需每次循环取出两数组中最小项即可排序,剩余的最大数还留在left或者right数组中需要通过concat合并过来
function mergeSort(arr) { //采用自上而下的递归方法
        var len = arr.length;
        if (len < 2) {
            return arr;
        }
        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) {
            // 不断比较left和right数组的第一项,小的取出存入res
            left[0] < right[0] ? result.push(left.shift()) : result.push(right.shift());
        }
        return result.concat(left, right);
    }

    console.log(mergeSort(a))

六、希尔排序

希尔排序(Shell Sort)是插入排序的一种。也称缩小增量排序,是直接插入排序算法的一种更高效的改进版本。希尔排序是非稳定排序算法。希尔排序是把记录按下标的一定增量分组,对每组使用直接插入排序算法排序;随着增量逐渐减少,每组包含的关键词越来越多,当增量减至1时,整个文件恰被分成一组,算法便终止。


function shellSort(arr) {
        for (let i = Math.floor(arr.length); i > 0; i = Math.floor(i / 2)) {
            for (let j = i; j < arr.length; j++) {
                let k = j;
                let currentValue = arr[j];
                while (k - i >= 0 && arr[k - i] > currentValue) {
                    arr[k] = arr[k - i];
                    k = k - i;
                }
                arr[k] = currentValue
            }
        }
    }

    var arr = [3, 5, 7, 1, 4, 56, 12, 78, 25, 0, 9, 8, 42, 37];
    shellSort(arr)
    console.log(arr)
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351