高性能JavaScript一点小知识

想必大家都知道,JavaScrip是全栈开发语言,浏览器,手机,服务器端都可以看到JS的身影。 本文会分享一些高效的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进行开发了。

时间: 2024-08-25 18:05:55

高性能JavaScript一点小知识的相关文章

C++ 中有关const引用的一点小知识

在读<C++ Primer>时,发现之前对const的概念不是很清晰,只知道如何去使用,于是翻开const引用部分又阅读了一遍,发现有两点自己要注意的地方 1.const限定的对象不可以初始化非const引用 ex. 1 const int src = 512; 2 const int &ok_dest = src; //ok: 引用和初始化对象都是const 3 int &err_dest = src; //error : 引用为非const 原因很简单, src 是不可以被

js调试的一点小知识

1.如果想要js代码被XHTML和HTML解析,就可以使用如下方式 <script type="text/javascript"> //<![CDATA[ function(){ …… } //]]> </script> 2.在调试器中可以通过callStack看到函数执行的过程. 3.控制台比较有用的函数 console.log():输出日志 console.trace():堆栈函数,可以查看指定函数的调用关系 clear():清除控制台中的log

javascript构造函数小知识

1.默认函数首字母大写 2.构造函数并没有显示返回任何东西.new 操作符会自动创建给定的类型并返回他们,当调用构造函数时,new会自动创建this对象,且类型就是构造函数类型. 3.也可以在构造函数中显示调用return.如果返回的值是一个对象,它会代替新创建的对象实例返回.如果返回的值是一个原始类型,它会被忽略,新创建的实例会被返回. function Person(name) { this.name = name; } var p1 = new Person('John'); 等同于: f

http一点小知识

1.什么是Uri? 统一资源标识符 uniform resource identifier 2.什么是URL? 统一资源定位符 uniform resource locator 是一种特殊的uri 3.什么是http? HTTP是一种基于请求-响应模式的.无状态的协议 上图是客户请求一次http的过程 请求报文: 响应报文: 4.http请求由几个部分组成? 请求行+消息报头+请求正文 请求行(Method + RequestUri + http+version + CRLF) eg : GET

Android的一点小知识

1.如何隐藏标题栏 在Android的MainActivity中的onCreate函数中加入requestWindowFeature(Window.FEATURE_NO_TITLE);一定要加在setContentView前面. 2.logcat中的级别 log.v()----------->verbose(一些细小的日志); log.d()----------->debug(用于调试程序): log.i()------------>info(打印比较重要的信息,帮助分析用户行为); l

一点小知识的累计

C++一个减号和一个大于号是什么意思? "->"相当于结构体变量后的".",只不过"->"的左边是指针,"."左边是普通变量. 如:typedef struct{ int x; int y;} Point;struct Point a;struct Point *p;a.x=4;a.y=5;p=&a;printf("%d %d",p->x,p->y); ===========

每天一点小知识002-玩kinectwrapper的一些记录

我的unity版本是4.6 kinect一代 kinectsdk 1.7 之前用csdn所谓的1.8更新版怎么都读不出彩色图和深度图 找了wiki百科  说是1.7更新了这个bug 换回下载的kinect wrapper1.7才读出来了,(估计那位1.8更新版用的不是1.7更新的..) 打开kinectExample场景,读取深度图很简单,DisplayDepth脚本就是了,不清楚是不是我这个一代驱动有点问题,有时显示不出图像,将kinect拔出再插上又行了(omg...) DisplayDep

关于可变数组的一点小知识

最近在使用iOS的可变数组时出现了一个错误,报错信息为: reason: '-[__NSCFArray removeObjectAtIndex:]: mutating method sent to immutable object' 字面的意思就是可变数组的方法发送给了一个不可变的对象. 声明和初始化的代码如下: //声明 NSMutableArray *dataArray; //初始化 dataArray = [NSMutableArray array]; 通过分析dataArray,发现只有

一点小知识

cmd打开注册表 regedit HKEY_USERS包含所有加载的用户配置文件 HKEYCURRENT_USERS当前登录用户的配置文件 HKEY_CLASSES_ROOT包含所有已祖册的文件类型,OLE等信息 HKEYCURRENT_CONFIG启动时系统硬件配置文件 HKEYLOCAL_MACHINE配置信息,包括硬件和软件设置 破解Windows密码:C:\Windows\System32\configSAM SYSTEM 文件 原文地址:https://www.cnblogs.com/