学习笔记-JavaScript性能优化

2021-03-22  本文已影响0人  _咻咻咻咻咻

内存管理

JavaScript中的内存管理:JavaScript中没有专门的API来操作内存空间,那么它是怎么申请使用并且释放的呢?

image.png

垃圾回收

Javascript中的垃圾:

JavaScript执行引擎把垃圾占据的对象空间进行回收,这个过程就是垃圾回收

JavaScript中的可达对象:

GC算法

常见的GC算法:

引用计数

实现原理:在内部通过一个引用计数器来维护当前对象的引用数,判断当前引用数是否为0来决定它是不是一个垃圾对象。如果是,执行引擎就将它回收并释放。

优点

缺点

标记清除

实现原理:分标记和清除两个阶段。第一个阶段:遍历所有对象找到活动对象进行标记;第二阶段:遍历所有对象把没有标记的对象进行清除(在这个阶段会把第一阶段的标记给抹掉,便于GC下次正常工作)。

优点

缺点

标记整理

优点

缺点

V8

V8 垃圾回收策略

V8 采用分代回收的思想,将内存空间一分为二,并针对不同对象采用不同算法。

image.png

V8常用的GC算法:

回收新生代对象

新生代对象指的是存活时间较短的对象

回收细节说明:

回收老生代对象

老生代对象就是指存活时间较长的对象:全局变量,闭包中放置的变量数据。

增量标记

image.png

当垃圾回收的时候,是会阻塞 JavaScript 代码的执行,所以会有空档期。所谓的增量标记就是将一整段的垃圾回收操作拆分成多个小步,组合着完成整个回收,从而是程序执行和垃圾回收交替进行,这样带来的时间消耗更加合理。

老生代 vs. 新生代

Performance 工具介绍

为什么使用Performance?

使用步骤:

监控内存的几种方式

内存问题的外在表现:

界定内存问题的标准:

监控内存的几种方式:

任务管理器监控内存

shift + esc 调出任务管理器,操作页面后观察任务管理器的内存占用空间(DOM 节点所占用的内存)和JavaScript占用内存。

image.png

如果内存一直增长就是有问题的,但是任务管理器无法定位问题。

Timeline 记录内存

image.png

在性能选项录制停止后,可以在这里查看内存走势。

堆快照查找分离 DOM

什么是分离DOM?
界面元素都是DOM节点,存活在DOM树上,DOM 节点存在几种形态:

分离DOM在界面上是看不见的,但是在内存里占据了空间,这就是一种内存泄漏。可以通过堆快照的功能找到这些分离DOM。

image.png image.png

判断是否存在频繁GC

确定频繁的垃圾回收:

代码优化

如何精准测试 JavaScript 性能?

慎用全局变量

为什么要慎用?

eg:

// 代码 1
var i, str = ''
for (i = 0; i < 1000; i++) {
  str += i
}

// 代码 2
for (let i = 0; i < 1000; i++) {
  let str = ''
  str += i
}

将这段代码进行测试。

image.png

可以发现使用局部变量性能要好很多。

缓存全局变量

将使用中无法避免的全局变量缓存到局部。比如 document。

image.png

通过原型新增方法

在原型对象上新增实例对象需要的方法。

image.png

避开闭包陷阱

闭包的特点:

// 闭包
function foo() {
  var name = 'lg'
  function fn() {
    console.log(name)
  }
  return fn
}
var a = foo()
a()

我们根据一个例子来解释闭包陷阱

<button id="btn">Add</button>

<script>
  // 代码 1
  function foo() {
    var el = document.getElementById('btn')
    el.onclick = function () {
      console.log(el.id)
    }
  }
  foo()

  // 代码 2
  function foo() {
    var el = document.getElementById('btn')
    el.onclick = function () {
      console.log(el.id)
    }
    el = null
  }
  foo()
</script>

上面例子中,foo 函数中将 document.getElementById('btn') 的引用给了 el,但是这个 btn 元素,本身就有一个dom元素的引用,在这里赋值,就变成了两个引用。在引用计数法中,如果dom元素被删除,这个dom元素的引用减一,但是这个dom元素还存在一个引用,这个内存就会无法释放。所以可以通过将 el 置为 null 的方法,手动释放这个内存,以防大量的这种错误使用造成内存泄漏。

避免属性访问方法使用

JavaScript中的面向对象

image.png

For循环优化 - 减少循环体中活动

将每次循环都要用到的变量抽离到外边。

<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<p class="btn">add</p>
<script>
  var aBtns = document.getElementsByClassName('btn')

  for (var i = 0; i < aBtns.length; i++) {
    console.log(i)
  }

  for (var i = 0, len = aBtns.length; i < len; i++) {
    console.log(i)
  }
</script>

上面两个for循环,一个将直接用 i < aBtns.length, 一个将 aBtns.length 存到变量 len, 再用 i<len,将两者进行对比,可以看出,将长度先存到变量再进行比较性能要好一些。

image.png

采用最优循环方式

image.png image.png image.png image.png image.png image.png image.png

可以看出如果只是简单的遍历,while在数组大小小于5000时是最快的,但是大于5000就特别慢;而for永远是最慢的;for...in在数组越大它执行的越快;forEach则比较稳定,相对速度都比较快。
数组较小时:while > forEach > for...in > for
数组较大时:for...in > forEach > while > for
使用 forEach 的性能最好,其次是 for...in,最后才是 for 循环。

文档碎片 优化节点添加

节点的添加操作必然会有回流和重绘,而这两个对性能的消耗是非常大的。那我们怎么进行一个最优操作呢?

// 使用普通方法创建节点 p, 将 p 节点加到文档末尾
for (var i = 0; i < 10; i++) {
  var oP = document.createElement('p')
  oP.innerHTML = i
  document.body.appendChild(oP)
}

// 创建一个文档碎片容器,将 p 节点放入文档碎片容器的末尾
// 最后再将文档最偏容器放入 body 的末尾
const fragEle = document.createDocumentFragment()
for (var i = 0; i < 10; i++) {
  var oP = document.createElement('p')
  oP.innerHTML = i
  fragEle.appendChild(oP)
}
document.body.appendChild(fragEle)

将两段代码进行性能对比后发现,通过“文档碎片容器”的方式添加节点性能更好。

image.png

克隆 优化节点操作

<!-- 页面中有一个 p 标签,要创建一个和它一样的标签 -->
<p id="box1">old</p>
<script>
  // 使用创建节点并添加的方式
  for (var i = 0; i < 3; i++) {
    var oP = document.createElement('p')
    oP.innerHTML = i
    document.body.appendChild(oP)
  }

  // 使用克隆 p 标签的方式
  var oldP = document.getElementById('box1')
  for (var i = 0; i < 3; i++) {
    var newP = oldP.cloneNode(false)
    newP.innerHTML = i
    document.body.appendChild(newP)
  }
</script>
image.png

减少判断层级

function doSomeThing(part, chapter) {
  const parts = ['ES2016', '工程化', 'Vue', 'React', 'Node']
  if (part) {
    if (parts.includes(part)) {
      console.log('属于当前课程')
      if (chapter > 5) {
        console.log('您需要提供 VIP 身份')
      }
    }
  } else {
    console.log('请确认模块信息')
  }
}

function doSomeThing(part, chapter) {
  const parts = ['ES2016', '工程化', 'Vue', 'React', 'Node']
  if (!part) {
    console.log('请确认模块信息')
    return
  }
  if (!parts.includes(part)) return
  console.log('属于当前课程')
  if (chapter > 5) {
    console.log('您需要提供 VIP 身份')
  }
}

以上两种方式,第一种if嵌套很多,第二种减少了嵌套层级,根据结果可以看出性能更好。

image.png

减少作用域链查找层级

var name = 'aaa'
function foo() {
  name = 'bbb'  // 这里的name是属于全局的
  function baz() {
    var age = 38
    console.log(age)
    console.log(name)
  }
  baz()
}
foo()


var name = 'aaa'
function foo() {
  var name = 'bbb'
  function baz() {
    var age = 38
    console.log(age)
    console.log(name)
  }
  baz()
}
foo()
image.png

可以看出,变量在作用域中的位置离的越近,查找的越快。不过这只是从时间上考虑的,如果从空间上考虑,第一种方法的name只占用了一个内存空间,而第二种两个name占了两个空间,占用内存相对较大。具体项目中要根据需求判断考虑空间还是时间。

减少数据读取次数

<div id="skip" class="skip"></div>
<script>
  var oBox = document.getElementById('skip')
  // function hasEle(ele, cls) {
  //   return ele.className == cls
  // }

  function hasEle(ele, cls) {
    var clsName = ele.className
    return clsName == cls
  }
  console.log(hasEle(oBox, 'skip')) // true
</script>

使用 var clsName = ele.className 可以减少 className 的读取次数,提高性能。

字面量与构造式

image.png image.png image.png

可以看出,直接通过字面量定义的方式,比通过 new 的方式要快很多,尤其是在普通数据类型特别明显,Object 类型虽然相差不多,但也是字面量的方式要快的,因为 new 的方式实际上是去调用了函数。字面量的方式除了快,它书写也是很方便的。

减少生命及语句数

对于后续不需频繁使用的数据建议使用时获取,而不做缓存,减少对内存的消耗。

采用事件委托

上一篇 下一篇

猜你喜欢

热点阅读