JavaScript数组的常用算法

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。

欢迎大家去我的个人技术博客看看,点赞收藏注册的都是好人哦~

https://xiaowu.xyz

一、数组的常见算法

由于算法的性能要从时间复杂度和空间复杂度两个方面考虑,所以这里不做性能的研究,仅仅为了理解

1、冒泡排序:

  假设有数组[54, 68, 46, 75, 36, 20, 65, 11, 79, 45]

var list = [54, 68, 46, 75, 36, 20, 65, 11, 79, 45];   // 要排序的数组

   需求:从小到大排列数组

因为我们要把大的数放在后面,所以我们每个数都和右边的数比较,如果左边的数大于右边的数,就将两个数换位置

   循环一圈后,就能得到最大的值,放在最右边,每次循环一圈,就能得到相对最大的值放在最右边

    

第一轮  就得到最大值  j=0

    54 68  [54, 68, 46, 75, 36, 20, 65, 11, 79, 45]

    68 46  [54, 46, 68, 75, 36, 20, 65, 11, 79, 45]

    68 75  [54, 46, 68, 75, 36, 20, 65, 11, 79, 45]

    75 36  [54, 46, 68, 36, 75, 20, 65, 11, 79, 45]

    75 20  [54, 46, 68, 36, 20, 75, 65, 11, 79, 45]

    75 65  [54, 46, 68, 36, 20, 65, 75, 11, 79, 45]

    75 11  [54, 46, 68, 36, 20, 65, 11, 75, 79, 45]

    75 79  [54, 46, 68, 36, 20, 65, 11, 75, 79, 45]

    75 45  [54, 46, 68, 36, 20, 65, 11, 75, 45, 79]

    for (var i = 0; i < list.length - 1; i++) {  //第一次循环

        if (list[i] > list[i + 1]) {  // 68 46   //如果左边的值大于右边成立

            var temp = list[i];    // 设置一个临时变量,将左边的值先存起来

            list[i] = list[i + 1];     // 将右边的值赋给左边

            list[i + 1] = temp;    //  将临时变量中存的最初的左边的值给右边   完成一次交换位置

        }

    }

    console.log(list);

    

第二轮  得到  第二大的值  j=1

    54 46  [46, 54, 68, 36, 20, 65, 11, 75, 45, 79]

    54 68  [46, 54, 68, 36, 20, 65, 11, 75, 45, 79]

    68 36  [46, 54, 36, 68, 20, 65, 11, 75, 45, 79]

    68 20  [46, 54, 36, 20, 68, 65, 11, 75, 45, 79]

    68 65  [46, 54, 36, 20, 65, 68, 11, 75, 45, 79]

    68 11  [46, 54, 36, 20, 65, 11, 68, 75, 45, 79]

    68 75  [46, 54, 36, 20, 65, 11, 68, 75, 45, 79]

    75 45  [46, 54, 36, 20, 65, 11, 68, 45, 75, 79]

    75 79  [46, 54, 36, 20, 65, 11, 68, 45, 75, 79]  //多余一次   由于第一次已经取到最大的值了,所以最后一次不用比较

    for (var i = 0; i < list.length - 1; i++) {   //第二次循环

        if (list[i] > list[i + 1]) {  

            var temp = list[i];     // 设置一个临时变量,将左边的值先存起来

            list[i] = list[i + 1];       // 将右边的值赋给左边

            list[i + 1] = temp;      //  将临时变量中存的最初的左边的值给右边   完成一次交换位置

        }

    }

    console.log(list);

 

第三轮  得到 第三大的值  j=2

    45 54  [46, 54, 36, 20, 65, 11, 68, 45, 75, 79]

    54 36  [46, 36, 54, 20, 65, 11, 68, 45, 75, 79]

    54 20  [46, 36, 20, 54, 65, 11, 68, 45, 75, 79]

    54 65  [46, 36, 20, 54, 65, 11, 68, 45, 75, 79]

    65 11  [46, 36, 20, 54, 11, 65, 68, 45, 75, 79]

    65 68  [46, 36, 20, 54, 11, 65, 68, 45, 75, 79]

    65 45  [46, 36, 20, 54, 11, 65, 45, 68, 75, 79]

    65 75  [46, 36, 20, 54, 11, 65, 45, 68, 75, 79]  //多余

    75 79  [46, 36, 20, 54, 11, 65, 45, 68, 75, 79]  //多余      //多余两次   由于前两次已经取到最大的值和第二大的值了,所以最后两次不用比较

    ......

所以完整的冒泡排序

var list = [54, 68, 46, 75, 36, 20, 65, 11, 79, 45];   // 要排序的数组

for (var j = 0; j < list.length - 1; j++) {   //外层决定执行多少轮  因为最后一个元素没有下一个元素和他比较,所以 j要小于list.length - 1

        for (var i = 0; i < list.length - 1 - j; i++) {    // 内层从最左边开始比较,-j是为了性能优化,将多余的比较减掉

            if (list[i] > list[i + 1]) {  

                var temp = list[i];

                list[i] = list[i + 1];

                list[i + 1] = temp;

            }

        }

    }

    console.log(list);

递归写法:

    function maoPao(list, j) {

        for (var i = 0; i < list.length - 1; i++) {

            if (list[i] > list[i + 1]) {  // 68 46

                var temp = list[i];

                list[i] = list[i + 1];

                list[i + 1] = temp;

            }

        }

        console.log(list);

        if (j == 1) {

            return list;

        }

        return maoPao(list, j - 1);

    }

    var list = [54, 68, 46, 75, 36, 20, 65, 11, 79, 45];   // 要排序的数组

    maoPao(list, list.length); //10 

// 这里的能用循环还是最好用循环,这里只是为了练习递归

2、选择排序:

假设我们有数组[54, 68, 46, 75, 36, 20, 65, 11, 79, 45],我们要求从小到大的排列

var list = [54, 68, 46, 75, 36, 20, 65, 11, 79, 45];    

选择排序和冒泡排序不同,冒泡排序每轮循环都要换很多次位置,而选择排序每次循环只换一次位置。

从下标0开始,与其后所有元素相比较,找到后面最小的值,然后与其交换位置

然后再从下标1开始找,找到后面最小的值,然后与其交换位置

  var list = [54, 68, 46, 75, 36, 20, 65, 11, 79, 45];

    for (var j = 0; j < list.length - 1; j++) {//0

        var min = list[j];      // 把每次循环的次数对应的那个值保存下来

        var minIndex = j;  //把每次循环的次数对应的那个值对应的下标也保存下来

        for (var i = j + 1; i < list.length; i++) {

            if (min > list[i]) {     

                min = list[i];

                minIndex = i;

            }

        }

        // 内层for循环执行结束后,就一定能得到后面最小的那个值的下标和值

        // 将当前的值和后面最小的值交换顺序

        var temp = list[j];

        list[j] = list[minIndex];

        list[minIndex] = temp;

    }

console.log(list);

3、快速排序(二分排序):

这里利用递归来实现快速排序

① 选取待排序数组中其中一个数作为基数(这里选择了位置偏中间的数Math.floor(list.length / 2)),使midIndex等于基数的下标,将基数从数组中裁切下来,建立两个空数组,left数组和right数组

② 将数组中比基数小的数放到left数组中,比基数大的数放到rigth数组中。

③ 将基数左边的数组作为待排序数组,重复①步骤。

④ 将基数右边的数组作为待排序数组,重复①步骤。

⑤ 直到left和right的长度都小于2(数组长度为1或0),代表数组已经划分为最小单位(待排序数组长度小于等于1),即该部分排序完毕,无需继续分割数组。

    function quickSort(list) {

        if (list.length < 2) {

            return list;

        }

        var midIndex = Math.floor(list.length / 2);

        var mid = list.splice(midIndex, 1)[0];   //返回的是数组

        var left = [];

        var right = [];

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

            if (list[i] < mid) {

                left.push(list[i]);

            } else {

                right.push(list[i]);

            }

        }

        // console.log(left, mid, right);

        return quickSort(left).concat(mid, quickSort(right));

    }

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

推荐阅读更多精彩内容

  • 版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。 欢迎大家...
    小五丶_阅读 284评论 0 0
  • 一、常见排序算法一览: 时间复杂度: 是一个函数,它定量描述了该算法的运行时间。 空间复杂度:一个算法在运行过程中...
    夕望有你阅读 876评论 0 0
  • 夜凉如秋水,露白似凝霜。 时光倏尔远,人生一梦长。
    斓心澈瞳阅读 92评论 0 0
  • 地理考试的时候我悄悄问前桌,“死亡率怎么求?” 监考老师一个眼神杀了过来。 我赶紧换了一个语气,“有没...
    密啊玛斯阅读 139评论 0 1
  • 在这个各方面都飞速发展的时代,焦虑似乎无处不在。每天在地铁里、公交上、商场里、医院里、学校里,到处都可以...
    萌萌哒终结者阅读 502评论 2 2