资讯专栏INFORMATION COLUMN

10个优化JavaScript代码实用小tips

3403771864 / 401人阅读

  说道JavaScript的代码优化,就先要做的是准确的测试JavaScript的代码执行时间。简单来说就是采集大量的执行样本进行数学统计和分析,这里我们使用的是benchmark.js来检测代码的执行情况。

  首先我们需要在项目中安装依赖,代码如下:

  yarn add benchmark --save
  # 或者
  npm i benchmark --save

  然后我们写一个测试代码,如下所示: 

 const Benchmark = require('benchmark')
  const suite = new Benchmark.Suite()
  // 添加测试
  suite
  /**
  * add() 方法接受两个参数,其中第一个表示测试的名称,第二个表示测试的内容,他是一个函数*
  /
  .add('join1000', () => {
  new Array(1000).join(' ')
  })
  .add('join10000', () => {
  new Array(10000).join(' ')
  })
  // 添加时间监听
  .on('cycle', event => {
  // 打印执行时间
  console.log(String(event.target))
  })
  // 完成后执行触发的事件
  .on('complete', () => {
  console.log('最快的是:' + suite.filter('fastest').map('name'))
  })
  // 执行测试
  .run({ async: true })
  复制代码
  代码执行结果如下:
  // join1000 x 146,854 ops/sec ±1.86% (88 runs sampled)
  // join10000 x 16,083 ops/sec ±1.06% (92 runs sampled)
  // 最快的是:join1000

  在结果中,ops/sec表示的是每秒执行的次数,越大表示越好,每秒执行次数上下相差的百分比,最后括号中的内容表示共取样多少次。

  或者也可以使用JSBench.me工具进行替换,网站测试截图如下:

2.png

  上面可以看待join1000的性能更好一些。

  慎用全局变量  

      慎用全局变量,为什么要慎用呢?主要有以下几点:

  全局变量定义在全局执行上下文,是所有作用域链的顶端。每次查找的时候都从局部找到最顶端,在时间上会有所消耗。

  全局执行上下文一直存在于上下文的执行栈,直到程序退出,才会被销毁,内存空间浪费。

  如果某个局部作用域出现了同名的变量则会遮盖或者说污染全局变量 。

  下面我们就来写一段代码,看一下全局变量与布局变量在执行效率方面的差异,代码如下:

  ...
  suite
  .add('全局变量', () => {
  // 该函数内模拟全局作用域
  let i,
  str = ''
  for (i = 0; i < 1000; i++) {
  str += i
  }
  })
  .add('局部变量', () => {
  for (let i = 0, str = ''; i < 1000; i++) {
  str += i
  }
  })
  ...

  代码运行结果如下:

  全局变量 x 158,697 ops/sec ±1.05% (87 runs sampled)

  局部变量 x 160,697 ops/sec ±1.03% (90 runs sampled)

  最快的是:局部变量

  这些差异化不大,但能够感知全局变量比局部的性能更差一些。

  通过原型新增方法

  为构造函数增加实例对象需要的方法时,尽量使用原型的方式添加,而不是构造函数内部进行添加,我们可以看如下测试代码:

  ...
  suite
  .add('构造函数内部添加', () => {
  function Person() {
  this.sayMe = function () {
  return '一碗周'
  }
  }
  let p = new Person()
  })
  .add('原型方式内部添加', () => {
  function Person() {}
  Person.prototype.sayMe = function () {
  return '一碗周'
  }
  let p = new Person()
  })
  ...

  代码运行结果如下:

  构造函数内部添加 x 573,786 ops/sec ±1.97% (89 runs sampled)

  原型方式内部添加 x 581,693 ops/sec ±3.46% (80 runs sampled)

  最快的是:构造函数内部添加

  避免闭包中的内存泄露

  我们要避免闭包中的内存泄露主要是因为在闭包会使得函数的变量过程中都被保存在内存中,就会产生很大的内存,这就是不能滥用闭包缘由,内存过大就会造成页面阻塞,严重可能导致内存泄露。解决方法是,在退出函数之前,将不使用的局部变量全部删除(即将局部变量重新赋值为null)。

  避免使用属性访问方法

  在JavaScript中的对象中,避免使用一些属性访问方法,这是因为JavaScript中的所有属性都是外部可见的。

  示例代码如下: 

 ...
  suite
  .add('使用属性访问方法', () => {
  function Person() {
  this.name = '一碗周'
  this.getName = function () {
  return '一碗周'
  }
  }
  let p = new Person()
  let n = p.getName()
  })
  .add('不使用属性访问方法', () => {
  function Person() {
  this.name = '一碗周'
  }
  let p = new Person()
  let n = p.name
  })
  ...

  代码运行结果如下:

  使用属性访问方法 x 406,682 ops/sec ±2.33% (82 runs sampled)

  不使用属性访问方法 x 554,169 ops/sec ±2.03% (85 runs sampled)

  最快的是:不使用属性访问方法

  for循环优化

  我们在使用for循环时,可以将有些必要的数据进行缓存,就比如arr.length这种属性,不需要每次判断都获取一下,从而优化我们的代码。

  示例代码如下:

 

  ...
  suite
  .add('正序', () => {
  let arr = new Array(100)
  let str = ''
  for (let i = 0; i < arr.length; i++) {
  str += i
  }
  })
  .add('缓存', () => {
  let arr = new Array(100)
  let str = ''
  for (let i = arr.length; i; i--) {
  str += i
  }
  })
  .add('缓存的另一种写法', () => {
  let arr = new Array(100)
  let str = ''
  for (let i = 0, l = arr.length; i < l; i++) {
  str += i
  }
  })
  ...

  代码运行结果如下:

  正序 x 1,322,889 ops/sec ±1.36% (86 runs sampled)

  缓存 x 1,356,696 ops/sec ±0.70% (92 runs sampled)

  缓存的另一种写法 x 1,383,091 ops/sec ±0.70% (93 runs sampled)

  最快的是:缓存的另一种写法

  选择最优的循环方式

  我们现在常用的循环有forEach、for和for...in循环,这几种那个是性能最优的呢,测试代码如下:


  ...
  suite
  .add('forEach', () => {
  let arr = new Array(100)
  let str = ''
  arr.forEach(i => {
  str += i
  })
  })
  .add('for...in', () => {
  let arr = new Array(100)
  let str = ''
  for (i in arr) {
  str += i
  }
  })
  .add('for', () => {
  let arr = new Array(100)
  let str = ''
  for (let i = 0, l = arr.length; i < l; i++) {
  str += i
  }
  })
  ...

  代码运行结果如下:

  forEach x 4,248,577 ops/sec ±0.89% (86 runs sampled)

  for...in x 4,583,375 ops/sec ±1.15% (91 runs sampled)

  for x 1,343,871 ops/sec ±1.91% (88 runs sampled)

  最快的是:for...in

  由运行结果可以看出我们可以尽量使用for...in或者forEach循环,减少使用for循环。

  减少判断层级

  减少判断层级就是减少一些if语句的嵌套,如果是一些必要的条件我们可以通过单层if结合return直接跳出函数的执行,关于优化前与优化后的代码执行比对如下所示:


  ...
  /***
  接收两类文件,zip 和 rar*
  压缩包的大小限制为 10 兆*
  /
  suite
  .add('嵌套写法', () => {
  function uploadFile(suffix, size) {
  // 允许上传的后缀名
  const suffixList = ['.zip', '.rar']
  const M = 1024* 1024
  if (suffixList.includes(suffix)) {
  if (size <= 10* M) {
  return '下载成功'
  }
  }
  }
  uploadFile('.zip', 1* 1024* 1024)
  })
  .add('减少判断写法', () => {
  function uploadFile(suffix, size) {
  // 允许上传的后缀名
  const suffixList = ['.zip', '.rar']
  const M = 1024* 1024
  if (!suffixList.includes(suffix)) return
  if (size > 10* M) return
  return '下载成功'
  }
  uploadFile('.zip', 1* 1024* 1024)
  })
  ...

  代码运行结果如下:

  嵌套写法 x 888,445,014 ops/sec ±2.48% (88 runs sampled)

  减少判断写法 x 905,763,884 ops/sec ±1.35% (92 runs sampled)

  最快的是:减少判断写法,嵌套写法

  这方法差不大,但它并不适用嵌套的代码比普通代码更优一些。

  减少作用域链查找层级

  减少代码中作用域链的查找也是代码优化的一种方法,如下代码展示了两者的区别: 

 

 ...
  suite
  .add('before', () => {
  var name = '一碗粥'
  function sayMe() {
  name = '一碗周'
  function print() {
  var age = 18
  return name + age
  }
  print()
  }
  sayMe()
  })
  .add('after', () => {
  var name = '一碗粥'
  function sayMe() {
  var name = '一碗周' // 形成局部作用域
  function print() {
  var age = 18
  return name + age
  }
  print()
  }
  sayMe()
  })
  ...

  代码运行结果如下:

  before x 15,509,793 ops/sec ±7.78% (76 runs sampled)

  after x 17,930,066 ops/sec ±2.89% (83 runs sampled)

  最快的是:after

  上面代码只是为了展示区别,并没有实际意义。

  减少数据读取次数

  如果对象中的某个数据在一个代码块中使用两遍以上,这样的话将其进行缓存从而减少数据的读取次数来达到更优的一个性能,

  测试代码如下:


  ...
  var userList = {
  one: {
  name: '一碗周',
  age: 18,
  },
  two: {
  name: '一碗粥',
  age: 18,
  },
  }
  suite
  .add('before', () => {
  function returnOneInfo() {
  userList.one.info = userList.one.name + userList.one.age
  }
  returnOneInfo()
  })
  .add('after', () => {
  function returnOneInfo() {
  let one = userList.one
  one.info = one.name + one.age
  }
  returnOneInfo()
  })
  ...

  代码运行结果如下:

  before x 222,553,199 ops/sec ±16.63% (26 runs sampled)

  after x 177,894,903 ops/sec ±1.85% (88 runs sampled)

  最快的是:before

  字面量与构造式

  凡是可以使用字面量方式声明的内容,绝对是不可以使用构造函数的方式声明的,两者在性能方面相差甚远,代码如下:


  ...
  suite
  .add('before', () => {
  var str = new String('string')
  })
  .add('after', () => {
  var str = 'string'
  })
  ...

  代码运行结果如下:

  before x 38,601,223 ops/sec ±1.16% (89 runs sampled)

  after x 897,491,903 ops/sec ±0.92% (92 runs sampled)

  最快的是:after

      分享已完毕,请大家关注更多后续精彩内容。

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/128003.html

相关文章

  • 45实用JavaScript贴士、技巧和最佳实践

    摘要:翻译正文第一次声明变量时,请不要忘记使用关键字声明使用代替空字符串转成布尔值都为每行代码的末尾最好都加上个分号最好给对象都添加上构造函数在使用和尽量小心。 翻译介绍 翻译标题:45 Useful JavaScript Tips, Tricks and Best Practices 翻译来源:http://modernweb.com/2013/12/23/45-useful-java...

    Moxmi 评论0 收藏0
  • 前端资源系列(4)-前端学习资源分享&前端面试资源汇总

    摘要:特意对前端学习资源做一个汇总,方便自己学习查阅参考,和好友们共同进步。 特意对前端学习资源做一个汇总,方便自己学习查阅参考,和好友们共同进步。 本以为自己收藏的站点多,可以很快搞定,没想到一入汇总深似海。还有很多不足&遗漏的地方,欢迎补充。有错误的地方,还请斧正... 托管: welcome to git,欢迎交流,感谢star 有好友反应和斧正,会及时更新,平时业务工作时也会不定期更...

    princekin 评论0 收藏0
  • 1月份前端资源分享

    摘要:更多资源请文章转自月份前端资源分享视频前端技术论坛融合不可错过的迷你库测试框架实例教程为你详细解读请求头的具体含意解析的库如果要用前端框架,开发流程是怎样的与有什么区别正确使用的方法是什么流程图插件小如何让元素只能输入纯文本前端技术中 更多资源请Star:https://github.com/maidishike... 文章转自:https://github.com/jsfront...

    solocoder 评论0 收藏0
  • 程序员练级攻略(2018):前端性能优化和框架

    摘要:,谷歌给的一份性能指南和最佳实践。目前而言,前端社区有三大框架和。随后重点讲述了和两大前端框架,给出了大量的文章教程和相关资源列表。我认为,使用函数式编程方式,更加符合后端程序员的思路,而是更符合前端工程师习惯的框架。 showImg(https://segmentfault.com/img/bVbjQAM?w=1142&h=640); 这个是我订阅 陈皓老师在极客上的专栏《左耳听风》...

    VEIGHTZ 评论0 收藏0
  • 程序员练级攻略(2018):前端性能优化和框架

    摘要:,谷歌给的一份性能指南和最佳实践。目前而言,前端社区有三大框架和。随后重点讲述了和两大前端框架,给出了大量的文章教程和相关资源列表。我认为,使用函数式编程方式,更加符合后端程序员的思路,而是更符合前端工程师习惯的框架。 showImg(https://segmentfault.com/img/bVbjQAM?w=1142&h=640); 这个是我订阅 陈皓老师在极客上的专栏《左耳听风》...

    CoffeX 评论0 收藏0

发表评论

0条评论

3403771864

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<