《前端算法系列》如何让前端代码速度提高60倍来源: https://juejin.im/post/5d034e83e51d45773e418a69
文章插图
今天的问题从排序算法入手 , 来讲解如何根据业务需求 , 结合金典的算法 , 来实现js高性能开发 。情景
老板让小明给公司的20000+条数据排个序 , 但是由于排序的操作会频繁发生 , 如果操作执行的时间很慢 , 则会严重降低用户体验 , 听到这条噩耗后小明开始了代码 。
1.毫无违和感的排序算法 小明根据需求 , 思考了一会 , 写下了如下算法:
/** * max排序 * @param {*} arr* 耗时:760ms */ 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() }复制代码【如何让前端代码速度提高60倍】自信的小明陶醉在自己的算法中 , 准备测试一下性能 ,
/* * @Author: Mr Jiang.Xu* @Date: 2019-06-11 10:25:23* @Last Modified by: Mr Jiang.Xu * @Last Modified time: 2019-06-13 21:03:59 * @desc 测试函数执行的时间 */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.冒泡排序
小明不甘心 , 在网上查找相关资料后 , 写下了如下冒泡排序代码:
/** * 置换函数 * @param {源数组} arr* @param {原数组的A项} indexA* @param {原数组的B项} indexB*/ function swap(arr, indexA, indexB) { [arr[indexA], arr[indexB]] = [arr[indexB], arr[indexA]]; }/** * 原始冒泡排序 * @param {数组} arr* 耗时:377ms */ 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 , 完美 , 小明放到项目中测试 , 频繁排序还是会有点卡顿 , 能不能再优化一下呢? 思考许久之后 , 小明完善了冒泡排序:
/** * 利用索引优化后的冒泡排序 * @param {数组} arr* 耗时:350ms */ 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 , 但收益很小 。小明开始和自己过不去了 , 在维基百科上继续查找 , 最后发现了一个方法:
/** * 在每趟排序中进行正向和反向两遍冒泡 , * 一次可以得到两个最终值(最大和最小),* 从而使外排序趟数大概减少了一半 * @param {*} arr* 耗时:312ms */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一下 , 于是后又安利了如下算法:/** * 插入排序 -- 基础版 * @param {*} arr * 耗时:897ms */ 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,小明留下了没技术的泪水 。
最后小明拿出了这个看家本领 , 查到了二分搜索 , 最后改造后代码入下:/** * 改造二分查找,查找小于value且离value最近的值的索引 * @param {*} arr * @param {*} maxIndex * @param {*} value */ 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; } /** * 使用二分法来优化插入排序 * @param {*} arr * 耗时:86ms */ 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; } 复制代码
推荐阅读
- JavaScript 表单验证如何实现的?
- 传真机怎么接收传真 如何使用传真机
- 中国专家谈如何防御“国家级网络攻击”
- 懂这10道JS经典算法题,你就是前端大神
- Win10太占空间?一秒钟工夫让硬盘增容10GB
- 剖析5大自媒体平台,让你清楚知道自媒体变现技巧!
- 淘宝补单怎么补,多少合适? 淘宝如何正确补单
- 如何将KPI指标拆解为具体的工作任务?
- 冰柠檬红茶如何做 冰柠檬红茶的制作方法
- 坦洋工夫红茶如何储存 三大存储方法介绍