栏目分类:
子分类:
返回
名师互学网用户登录
快速导航关闭
当前搜索
当前分类
子分类
实用工具
热门搜索
名师互学网 > IT > 软件开发 > Web开发 > JavaScript

《前端算法系列》如何让前端代码速度提高60倍

JavaScript 更新时间: 发布时间: IT归档 最新发布 模块sitemap 名妆网 法律咨询 聚返吧 英语巴士网 伯小乐 网商动力

《前端算法系列》如何让前端代码速度提高60倍


今天的问题从排序算法入手,来讲解如何根据业务需求,结合金典的算法,来实现js高性能开发。

情景

老板让小明给公司的20000+条数据排个序,但是由于排序的操作会频繁发生,如果操作执行的时间很慢,则会严重降低用户体验,听到这条噩耗后小明开始了代码。

1.毫无违和感的排序算法
小明根据需求,思考了一会,写下了如下算法:

 function maxSort(arr) {
     let result = [...arr];
     for(let i=0,len=result.length; i< len; i++) {
 let minV = Math.min(...result.slice(i))
 let pos = result.indexOf(minV,i)
 result.splice(pos, 1)
 result.unshift(minV)
     }
     return result.reverse()
 }

自信的小明陶醉在自己的算法中,准备测试一下性能,


const testArr = require('./testArr');
module.exports = async function getFnRunTime(fn) {
    let len = testArr.length;
    let startTime = Date.now(), endTime;
    let result = await fn(testArr);
    endTime = Date.now();
    console.log(result);
    console.log(`total time:${endTime-startTime}ms`,
  'test array'length:' + len, 
  result.length
    );
}

运行该测试函数后,耗时760ms,小明觉得还不错,放到项目中后,第一次操作还好,连续操作了几次后,页面明显卡顿。。。(求此时小明心里的阴影面积)

2.冒泡排序

小明不甘心,在网上查找相关资料后,写下了如下冒泡排序代码:

 function swap(arr, indexA, indexB) {
    [arr[indexA], arr[indexB]] = [arr[indexB], arr[indexA]];
 }


 function bubbleSort1(arr) {
    for (let i = arr.length - 1; i > 0; i--) {
      for (let j = 0; j < i; j++) {
 if (arr[j] > arr[j + 1]) {
   swap(arr, j, j + 1);
 }
      }
    }
  
    return arr;
  }

测试后耗时377ms,完美,小明放到项目中测试,频繁排序还是会有点卡顿,能不能再优化一下呢?
思考许久之后,小明完善了冒泡排序:

 
function bubbleSort2(arr) {
    let i = arr.length - 1;

    while (i > 0) {
 let pos = 0;

 for (let j = 0; j < i; j++) {
 if (arr[j] > arr[j + 1]) {
     pos = j;
     swap(arr, j, j + 1);
 }
 }
 i = pos;
    }

    return arr;
}

根据缓存索引位置来提高排序性能,时间节约了20ms,但收益很小。小明开始和自己过不去了,在维基百科上继续查找,最后发现了一个方法:

function bubbleSort3(arr) {
    let start = 0;
    let end = arr.length - 1;
  
    while (start < end) {
      let endPos = 0;
      let startPos = 0;
      for (let i = start; i < end; i++) {
 if (arr[i] > arr[i + 1]) {
     endPos = i;
     swap(arr, i, i + 1);
 }
      }
      end = endPos;
      for (let i = end; i > start; i--) {
 if (arr[i - 1] > arr[i]) {
   startPos = i;  
   swap(arr, i - 1, i);
 }
      }
      start = startPos;
    }
  
    return arr;
  }

通过在每趟排序中进行正向和反向两遍冒泡,小明把时间又降低了38ms,不错~

再次推荐大家有事多上上维基百科,总有一款适合你。
####3.插入排序
在收入小规模胜利后,小明膨胀了,狂言要把排序时间降低到100ms一下,于是后又安利了如下算法:

  function insertionSort(arr) {
    for (let i = 1, len = arr.length; i < len; i++) {
      const temp = arr[i];
      let preIndex = i - 1;
  
      while (arr[preIndex] > temp) {
 arr[preIndex + 1] = arr[preIndex];
 preIndex -= 1;
      }
      arr[preIndex + 1] = temp;
    }
  
    return arr;
  }

897ms,小明留下了没技术的泪水。

最后小明拿出了这个看家本领,查到了二分搜索,最后改造后代码入下:

  function binarySearch1(arr, maxIndex, value) {
    let min = 0;
    let max = maxIndex;
    
    while (min <= max) {
      const m = Math.floor((min + max) / 2);
  
      if (arr[m] <= value) {
 min = m + 1;
      } else {
 max = m - 1;
      }
    }
  
    return min;
  }


function insertionSort1(arr) {
    for (let i = 1, len = arr.length; i < len; i++) {
 const temp = arr[i];
 const insertIndex = binarySearch1(arr, i - 1, arr[i]);

 for (let preIndex = i - 1; preIndex >= insertIndex; preIndex--) {
 arr[preIndex + 1] = arr[preIndex];
 }
 arr[insertIndex] = temp;
    }

    return arr;
}

完美,只用了86ms!小明激动的站了起来,还拍了下桌子,全然无视观众的眼光。

小明已经满足的不要不要的了,对86ms相当满意,老板也对他刮目想看。

4.希尔排序

难道就没有提升的余地了么?进过调查研究表明,是有更优的方案的:

function shellSort(arr) {
    const len = arr.length;
    let gap = Math.floor(len / 2);
  
    while (gap > 0) {
      // gap距离
      for (let i = gap; i < len; i++) {
 const temp = arr[i];
 let preIndex = i - gap;
  
 while (arr[preIndex] > temp) {
   arr[preIndex + gap] = arr[preIndex];
   preIndex -= gap;
 }
 arr[preIndex + gap] = temp;
      }
      gap = Math.floor(gap / 2);
    }
  
    return arr;
  }

耗时15ms,膜拜。
####5.归并排序

function concatSort(arr) {
  const len = arr.length;

  if (len < 2) { return arr; }

  const mid = Math.floor(len / 2);
  const left = arr.slice(0, mid);
  const right = arr.slice(mid);

  return concat(concatSort(left), concatSort(right));
}

function concat(left, right) {
  const result = [];

  while (left.length > 0 && right.length > 0) {
    result.push(left[0] <= right[0] ? left.shift() : right.shift());
  }

  return result.concat(left, right);
}

耗时30ms,也想当优秀。还有没有更快的方法呢?答案是有的,但是会涉及到比较高僧的数学知识,放弃吧,孩子。。。

接下来会推出更多优秀的算法,敬请期待哦~

更多推荐
  • 让你瞬间提高工作效率的常用js函数汇总(持续更新)
  • 如何用不到200行代码写一款属于自己的js类库
  • 记一次老项目中的跨页面通信问题和前端实现文件下载功能
  • 《前端实战》之变量提升,函数声明提升及变量作用域详解
  • 《前端实战总结》如何在不刷新页面的情况下改变URL
  • css3实战汇总(附源码)
  • 5分钟教你用nodeJS手写一个mock数据服务器
  • 前端组件/库打包利器rollup使用与配置实战
  • web性能优化的15条实用技巧
  • 快速掌握es6+新特性及es6核心语法盘点
转载请注明:文章转载自 www.mshxw.com
本文地址:https://www.mshxw.com/it/244046.html
我们一直用心在做
关于我们 文章归档 网站地图 联系我们

版权所有 (c)2021-2022 MSHXW.COM

ICP备案号:晋ICP备2021003244-6号