JS性能优化38条"军规",2019年呕心力做

前言

先自我介绍下,Sonia,女,资深前端开发一枚,现就任于某鱼技术部,负责主站性能优化和重构工做。掐指一算,从事前端开发整整12个年头了,不少朋友常常问到代码优化及性能调试,这里给你们总结了一下JS开发中的一些代码规范及优化的技巧和经验,此文谈不上精品之做,但绝对是匠心之做,诚心之做,但愿给前端开发路上的你一点帮助。javascript

1. 避免全局查找

在一个函数中会用到全局对象存储为局部变量来减小全局查找,由于访问局部变量的速度要比访问全局变量的速度更快些。前端

2. 定时器

若是针对的是不断运行的代码,不该该使用setTimeout,而应该是用setInterval,由于setTimeout每一次都会初始化一个定时器,而setInterval只会在开始的时候初始化一个定时器。java

3. 字符串链接

若是要链接多个字符串,应该少使用+=,如:编程

x+=a; x+=b; x+=c; 应该写成 x+= a + b + c; 而若是是收集字符串,好比屡次对同一个字符串进行+=操做的话,最好使用一个缓存,使用JavaScript数组来收集,最后使用join方法链接起来。数组

4. 避免with语句

和函数相似 ,with语句会建立本身的做用域,所以会增长其中执行的代码的做用域链的长度,因为额外的做用域链的查找,在with语句中执行的代码确定比外面执行的代码要慢,在能不使用with语句的时候尽可能不要使用with语句。浏览器

5. 数字转换成字符串

通常最好用"" + 1来将数字转换成字符串,虽然看起来比较丑一点,但事实上这个效率是最高的,性能上来讲:("" +) > String() > .toString() > new String()缓存

6. 浮点数转换成整型

不少人喜欢使用parseInt(),其实parseInt()是用于将字符串转换成数字,而不是浮点数和整型之间的转换,咱们应该使用Math.floor()或者Math.round()。安全

7. 各类类型转换

若是定义了toString()方法来进行类型转换的话,推荐显式调用toString(),由于内部的操做在尝试全部可能性以后,会尝试对象的toString()方法尝试可否转化为String,因此直接调用这个方法效率会更高性能优化

8. 多个类型声明

在JavaScript中全部变量均可以使用单个var语句来声明,这样就是组合在一块儿的语句,以减小整个脚本的执行时间,就如上面代码同样,上面代码格式也挺规范,让人一看就明了。微信

9. 插入迭代器

如var a=arr[i]; i++;前面两条语句能够写成var a=arr[i++]

10. 使用直接量

11. 使用DocumentFragment优化屡次append

一旦须要更新DOM,请考虑使用文档碎片来构建DOM结构,而后再将其添加到现存的文档中。

12. 使用一次innerHTML赋值代替构建dom元素

对于大的DOM更改,使用innerHTML要比使用标准的DOM方法建立一样的DOM结构快得多。

13. 经过模板元素clone,替代createElement

不少人喜欢在JavaScript中使用document.write来给页面生成内容。事实上这样的效率较低,若是须要直接插入HTML,能够找一个容器元素,好比指定一个div或者span,并设置他们的innerHTML来将本身的HTML代码插入到页面中。一般咱们可能会使用字符串直接写HTML来建立节点,其实这样作,1没法保证代码的有效性2字符串操做效率低,因此应该是用document.createElement()方法,而若是文档中存在现成的样板节点,应该是用cloneNode()方法,由于使用createElement()方法以后,你须要设置屡次元素的属性,使用cloneNode()则能够减小属性的设置次数——一样若是须要建立不少元素,应该先准备一个样板节点。

14. 使用firstChild和nextSibling代替childNodes遍历dom元素

15. 删除DOM节点

删除dom节点以前,必定要删除注册在该节点上的事件,无论是用observe方式仍是用attachEvent方式注册的事件,不然将会产生没法回收的内存。另外,在removeChild和innerHTML=’’两者之间,尽可能选择后者. 由于在sIEve(内存泄露监测工具)中监测的结果是用removeChild没法有效地释放dom节点。

16. 使用事件代理

任何能够冒泡的事件都不只仅能够在事件目标上进行处理,目标的任何祖先节点上也能处理,使用这个知识就能够将事件处理程序附加到更高的地方负责多个目标的事件处理,一样,对于内容动态增长而且子节点都须要相同的事件处理函数的状况,能够把事件注册提到父节点上,这样就不须要为每一个子节点注册事件监听了。另外,现有的js库都采用observe方式来建立事件监听,其实现上隔离了dom对象和事件处理函数之间的循环引用,因此应该尽可能采用这种方式来建立事件监听。

17. 重复使用的调用结果,事先保存到局部变量

18. 注意NodeList

最小化访问NodeList的次数能够极大的改进脚本的性能。

编写JavaScript的时候必定要知道什么时候返回NodeList对象,这样能够最小化对它们的访问,进行了对getElementsByTagName()的调用,获取了元素的childNodes属性,获取了元素的attributes属性,访问了特殊的集合,如document.forms、document.images等等,要了解了当使用NodeList对象时,合理使用会极大的提高代码执行速度。

19. 优化循环

可使用下面几种方式来优化循环。

减值迭代

大多数循环使用一个从0开始、增长到某个特定值的迭代器,在不少状况下,从最大值开始,在循环中不断减值的迭代器更加高效。

简化终止条件

因为每次循环过程都会计算终止条件,因此必须保证它尽量快,也就是说避免属性查找或者其它的操做,最好是将循环控制量保存到局部变量中,也就是说对数组或列表对象的遍历时,提早将length保存到局部变量中,避免在循环的每一步重复取值。

简化循环体

循环体是执行最多的,因此要确保其被最大限度的优化

使用后测试循环

在JavaScript中,咱们可使用for(;;),while(),for(in)三种循环,事实上,这三种循环中for(in)的效率极差,由于他须要查询散列键,只要能够,就应该尽可能少用。for(;;)和while循环,while循环的效率要优于for(;;),多是由于for(;;)结构的问题,须要常常跳转回去。

最经常使用的for循环和while循环都是前测试循环,而如do-while这种后测试循环,能够避免最初终止条件的计算,所以运行更快。

20. 展开循环

当循环次数是肯定的,消除循环并使用屡次函数调用每每会更快。

21. 避免双重解释

尽可能少使用eval函数

若是要提升代码性能,尽量避免出现须要按照JavaScript解释的字符串,也就是尽可能少使用eval函数, 使用eval至关于在运行时再次调用解释引擎对内容进行运行,须要消耗大量时间,并且使用Eval带来的安全性问题也是不容忽视的。

不要使用Function构造器

不要给setTimeout或者setInterval传递字符串参数

22. 缩短否认检测

23. 条件分支

将条件分支,按可能性顺序从高到低排列:能够减小解释器对条件的探测次数,在同一条件子的多(>2)条件分支时,使用switch优于if:switch分支选择的效率高于if,在IE下尤其明显。4分支的测试,IE下switch的执行时间约为if的一半,使用三目运算符替代条件分支。

24. 使用常量

重复值:任何在多处用到的值都应该抽取为一个常量,用户界面字符串:任何用于显示给用户的字符串,都应该抽取出来以方便国际化,URLs:在Web应用中,资源位置很容易变动,因此推荐用一个公共地方存放全部的URL,任意可能会更改的值:每当你用到字面量值的时候,你都要问一下本身这个值在将来是否是会变化,若是答案是“是”,那么这个值就应该被提取出来做为一个常量。

25. 避免与null进行比较

因为JavaScript是弱类型的,因此它不会作任何的自动类型检查,因此若是看到与null进行比较的代码,尝试使用如下技术替换,若是值应为一个引用类型,使用instanceof操做符检查其构造函数,若是值应为一个基本类型,做用typeof检查其类型,若是是但愿对象包含某个特定的方法名,则使用typeof操做符确保指定名字的方法存在于对象上。

26. 避免全局量

全局变量应该所有字母大写,各单词之间用_下划线来链接。尽量避免全局变量和函数,尽可能减小全局变量的使用,由于在一个页面中包含的全部JavaScript都在同一个域中运行。因此若是你的代码中声明了全局变量或者全局函数的话,后面的代码中载入的脚本文件中的同名变量和函数会覆盖掉(overwrite)你的。

27. 尊重对象的全部权

由于JavaScript能够在任什么时候候修改任意对象,这样就能够以不可预计的方式覆写默认的行为,因此若是你不负责维护某个对象,它的对象或者它的方法,那么你就不要对它进行修改,具体一点就是说:

  • 不要为实例或原型添加属性

  • 不要为实例或者原型添加方法

  • 不要重定义已经存在的方法

  • 不要重复定义其它团队成员已经实现的方法,永远不要修改不是由你全部的对象,你能够经过如下方式为对象建立新的功能。

  • 建立包含所需功能的新对象,并用它与相关对象进行交互

  • 建立自定义类型,继承须要进行修改的类型,而后能够为自定义类型添加额外功能

28. 循环引用

若是循环引用中包含DOM对象或者ActiveX对象,那么就会发生内存泄露。内存泄露的后果是在浏览器关闭前,即便是刷新页面,这部份内存不会被浏览器释放。简单的循环引用:

可是一般不会出现这种状况。一般循环引用发生在为dom元素添加闭包做为expendo的时候。

init在执行的时候,当前上下文咱们叫作context。这个时候,context引用了el,el引用了function,function引用了context。这时候造成了一个循环引用。

下面2种方法能够解决循环引用:

  • 置空dom对象

将el置空,context中不包含对dom对象的引用,从而打断循环应用。若是咱们须要将dom对象返回,能够用以下方法:

  • 构造新的context

把function抽到新的context中,这样,function的context就不包含对el的引用,从而打断循环引用。

29. 经过javascript建立的dom对象,必须append到页面中

IE下,脚本建立的dom对象,若是没有append到页面中,刷新页面,这部份内存是不会回收的!

30. 释放dom元素占用的内存

将dom元素的innerHTML设置为空字符串,能够释放其子元素占用的内存。在rich应用中,用户也许会在一个页面上停留很长时间,可使用该方法释放积累得愈来愈多的dom元素使用的内存。

31. 释放javascript对象

在rich应用中,随着实例化对象数量的增长,内存消耗会愈来愈大。因此应当及时释放对对象的引用,让GC可以回收这些内存控件。 对象:obj = null 对象属性:delete obj.myproperty 数组item:使用数组的splice方法释放数组中不用的item

32. 避免string的隐式装箱

对string的方法调用,好比'xxx'.length,浏览器会进行一个隐式的装箱操做,将字符串先转换成一个String对象。推荐对声明有可能使用String实例方法的字符串时,采用以下写法:

var myString = new String('Hello World');

33. 性能方面的注意事项

1. 尽可能使用原生方法

2. switch语句相对if较快

经过将case语句按照最可能到最不可能的顺序进行组织

3. 位运算较快

当进行数字运算时,位运算操做要比任何布尔运算或者算数运算快

4. 巧用||和&&布尔运算符

34. 避免错误应注意的地方

每条语句末尾须加分号

在if语句中,即便条件表达式只有一条语句也要用{}把它括起来,以避免后续若是添加了语句以后形成逻辑错误。

使用+号时需谨慎

JavaScript 和其余编程语言不一样的是,在 JavaScript 中,'+'除了表示数字值相加,字符串相链接之外,还能够做一元运算符用,把字符串转换为数字。于是若是使用不当,则可能与自增符'++'混淆而引发计算错误。

使用return语句须要注意

一条有返回值的return语句不要用()括号来括住返回值,若是返回表达式,则表达式应与return关键字在同一行,以免压缩时,压缩工具自动加分号而形成返回与开发人员不一致的结果。

34. ==和===的区别

避免在if和while语句的条件部分进行赋值,如if (a = b),应该写成if (a == b),可是在比较是否相等的状况下,最好使用全等运行符,也就是使用===和!==操做符会相对于==和!=会好点。==和!=操做符会进行类型强制转换。

35. 不要使用生偏语法

不要使用生偏语法,写让人迷惑的代码,虽然计算机可以正确识别并运行,可是晦涩难懂的代码不方便之后维护。

36. 函数返回统一类型

虽然JavaScript是弱类型的,对于函数来讲,前面返回整数型数据,后面返回布尔值在编译和运行均可以正常经过,但为了规范和之后维护时容易理解,应保证函数应返回统一的数据类型。

37. 什么时候用单引号,什么时候用双引号

虽然在JavaScript当中,双引号和单引号均可以表示字符串, 为了不混乱,咱们建议在HTML中使用双引号,在JavaScript中使用单引号,但为了兼容各个浏览器,也为了解析时不会出错,定义JSON对象时,最好使用双引号。

38. 老是检查数据类型

要检查你的方法输入的全部数据,一方面是为了安全性,另外一方面也是为了可用性。用户随时随地都会输入错误的数据。这不是由于他们蠢,而是由于他们很忙,而且思考的方式跟你不一样。用typeof方法来检测你的function接受的输入是否合法。

39. 结束语

  • 永远不要忽略代码优化工做,重构是一项从项目开始到结束须要持续的工做,只有不断的优化代码才能让代码的执行效率愈来愈好!

40. 个人公开课

最近加班加点录制了一套JS系列视频,涵盖了JS基础、JQ基础、JS高级、JS实战四个部分,你们能够报名学习下,都是免费的,后续我还会录制一套MVVM系列的视频,报名下面的课程后,你们也能够加我微信,有什么前端技术问题,均可以随时跟我讨论的。

JS系列视频教程地址点击学习>>>

个人微信(消息备注"掘金",谢谢!):

相关文章
相关标签/搜索