你不知道的高性能JAVASCRIPT


Posted in Javascript onJanuary 18, 2016

本文会分享一些高效的JavaScript的最佳实践,提高大家对JS的底层和实现原理的理解。

数据存储

计算机学科中有一个经典问题是通过改变数据存储的位置来获得最佳的读写性能,在JavaScript中,数据存储的位置会对代码性能产生重大影响。 ? 能使用{}创建对象就不要使用new Object,能使用[]创建数组就不要使用new Array。JS中字面量的访问速度要高于对象。 ? 变量在作用域链中的位置越深,访问所需实践越长。对于这种变量,可以通过缓存使用局部变量保存起来,减少对作用域链访问次数 ? 使用点表示法(object.name)和操作符(object[name])操作并没有太多区别,只有Safari会有区别,点始终更快

循环

在JS中常见的循环有下面几种:

for(var i = 0; i < 10; i++) { // do something} 
for(var prop in object) { // for loop object}  
[1,2].forEach(function(value, index, array) { // 基于函数的循环})

毋庸质疑,第一种方式是原生的,性能消耗最低的,速度也最快。第二种方式for-in每次迭代都回产生更多的开销(局部变量),它的速度只有第一种的1/7 第三种方式明显提供了更便利的循环方式,但是他的速度只有普通循环的1/8。所以可以根据自己项目情况,选择合适的循环方式。

事件委托

试想一下页面上每一个A标签添加一个事件,我们会不会给每一个标签都添加一个onClick呢。 当页面中存在大量元素都需要绑定同一个事件处理的时候,这种情况可能会影响性能。每绑定一个事件都加重了页面或者是运行期间的负担。对于一个富前端的应用,交互重的页面上,过多的绑定会占用过多内存。 一个简单优雅的方式就是事件委托。它是基于事件的工作流:逐层捕获,到达目标,逐层冒泡。既然事件存在冒泡机制,那么我们可以通过给外层绑定事件,来处理所有的子元素出发的事件。

document.getElementById('content').onclick = function(e) { 
  e = e || window.event;  
  var target = e.target || e.srcElement;  //如果不是 A标签,我就退出  
  if(target.nodeNmae !=== 'A') { return }  //打印A的链接地址  
  console.log(target.href) }

重绘与重排

浏览器下载完HTMl,CSS,JS后会生成两棵树:DOM树和渲染树。 当Dom的几何属性发生变化时,比如Dom的宽高,或者颜色,position,浏览器需要重新计算元素的几何属性,并且重新构建渲染树,这个过程称之为重绘重排。

bodystyle = document.body.style; 
bodystyle.color = red; 
bodystyle.height = 1000px; 
bodystyke.width = 100%;

上述方式修改三个属性,浏览器会进行三次重排重绘,在某些情况下,减少这种重排可以提高浏览器渲染性能。 推荐方式如下,只进行一次操作,完成三个步骤:

bodystyle = document.body.style; 
bodystyle.cssText 'color:red;height:1000px;width:100%';

JavaScript加载

IE8,Firefox3.5,Chrome2都允许必行下载JavaScript文件。所以<script>不会阻塞其他标签下载。 遗憾的是,JS下载过程依然会阻塞其他资源的下载,比如图片。尽管最新的浏览器通过允许并行下载提高了性能,但是脚本阻塞任然是一个问题。 因此,推荐将所有的<script>标签放在<body>标签的底部,以尽量减少对整个页面渲染的影响,避免用户看到一片空白

JS文件高性能部署

既然大家已经知道多个<script>标签会影响页面渲染速度,那么就不难理解“减少页面渲染所需的HTTP”是网站提速的一条经典法则。 所以,在产品环境下合并所有的JS文件会减少请求数,从而加快页面渲染速度。 除了合并JS文件,我们还可以压缩JS文件。压缩是指将文件中与运行无关的部分进行剥离。剥离内容包括空白字符,和注释。改过程通常可以将文件大小减半。 还有一些压缩工具会将局部变量的长度减小,比如:

var myName = "foo" + "bar"; 
//压缩后变成 
var a = "foobar";

缓存JS文件

缓存HTTP组件能极大提高网站回访的用户体验。Web服务器通过“Expires HTTP响应头”来告诉客户端一个资源应该缓存多长时间。当然,缓存也有自己的缺陷: 当应用升级时,你需要确保用户下载到最新的静态内容。这个问题可以通过改变静态资源的文件名来解决。 你可能在产品环境看到浏览器引用jsapplication-20151123201212.js,这种就是以时间戳的方式保存新的JS文件,从而解决缓存不更新问题。

总结

当然,高效的JS不仅仅只有这些可以改进的地方,如果能够减少一些性能的损耗,我们就能更高效的使用JavaScript进行开发了。

你不知道的高性能JAVASCRIPT,现在都知道了吧!

Javascript 相关文章推荐
许愿墙中用到的函数
Oct 07 Javascript
Date对象格式化函数代码
Jul 17 Javascript
AngularJS入门教程之AngularJS表达式
Apr 18 Javascript
AngularJS延迟加载html template
Jul 27 Javascript
Vue.js 表单校验插件
Aug 14 Javascript
基于angular中的重要指令详解($eval,$parse和$compile)
Oct 21 Javascript
如何制作幻灯片(代码分享)
Jan 06 Javascript
js正则相关知识点专题
May 10 Javascript
JavaScript简单实现的仿微博留言功能示例
Jan 17 Javascript
vue动态添加路由addRoutes之不能将动态路由存入缓存的解决
Feb 19 Javascript
Django模板继承 extend标签实例代码详解
May 16 Javascript
JavaScript面向对象程序设计中对象的定义和继承详解
Jul 29 Javascript
Ionic实现仿通讯录点击滑动及$ionicscrolldelegate使用分析
Jan 18 #Javascript
jQuery实现多级联动下拉列表查询框
Jan 18 #Javascript
jquery validate表单验证的基本用法入门
Jan 18 #Javascript
jQuery添加删除DOM元素方法详解
Jan 18 #Javascript
AngularJS仿苹果滑屏删除控件
Jan 18 #Javascript
jQuery easyui的validatebox校验规则扩展及easyui校验框validatebox用法
Jan 18 #Javascript
jquery.validate 自定义验证方法及validate相关参数
Jan 18 #Javascript
You might like
PHP批量生成静态HTML的简单原理和方法
2014/04/20 PHP
php实现无限级分类
2014/12/24 PHP
laravel 使用auth编写登录的方法
2019/09/30 PHP
Yii使用DbTarget实现日志功能的示例代码
2020/07/21 PHP
jquery 将disabled的元素置为enabled的三种方法
2009/07/25 Javascript
jquery的选择器的使用技巧之如何选择input框
2013/09/22 Javascript
再谈Jquery Ajax方法传递到action(补充)
2014/05/12 Javascript
jQuery统计指定子元素数量的方法
2015/03/17 Javascript
Web前端新人笔记之jquery入门心得(新手必看)
2016/05/17 Javascript
js注入 黑客之路必备!
2016/09/14 Javascript
JS框架之vue.js(深入三:组件1)
2016/09/29 Javascript
使用vue-router为每个路由配置各自的title
2018/07/30 Javascript
vue更改数组中的值实例代码详解
2020/02/07 Javascript
浅谈vue中resetFields()使用注意事项
2020/08/12 Javascript
浅谈vant组件Picker 选择器选单选问题
2020/11/04 Javascript
Python异常学习笔记
2015/02/03 Python
python实现从网络下载文件并获得文件大小及类型的方法
2015/04/28 Python
浅析Python中的赋值和深浅拷贝
2017/08/15 Python
基于Python os模块常用命令介绍
2017/11/03 Python
Python3中类、模块、错误与异常、文件的简易教程
2017/11/20 Python
如何在Django配置文件里配置session链接
2019/08/06 Python
Django中使用haystack+whoosh实现搜索功能
2019/10/08 Python
keras训练浅层卷积网络并保存和加载模型实例
2020/07/02 Python
CSS3实现瀑布流布局与无限加载图片相册的实例代码
2016/12/22 HTML / CSS
用CSS3实现无限循环的无缝滚动的实例代码
2017/07/04 HTML / CSS
瑞典时尚服装购物网站:Miinto.se
2017/10/30 全球购物
Kivari官网:在线购买波西米亚服装
2018/10/29 全球购物
Java中有几种类型的流?JDK为每种类型的流提供了一些抽象类以供继承,请说出他们分别是哪些类?
2012/05/30 面试题
创业计划书如何吸引他人眼球
2014/01/10 职场文书
行政助理的岗位职责
2014/02/18 职场文书
大学生求职信范文
2014/05/24 职场文书
民间借贷纠纷起诉书
2015/08/03 职场文书
厉行节约工作总结
2015/08/12 职场文书
情侣餐厅的创业计划书范本!
2019/07/26 职场文书
详解MySQL连接挂死的原因
2021/05/18 MySQL
CSS中妙用 drop-shadow 实现线条光影效果
2021/11/11 HTML / CSS