JavaScript的妙与乐(一)之 函数优化
JavaScript的妙与乐系列文章主要是展示一些JavaScript上面比较好玩一点的特性和一些有用的技巧,里面很多内容都是我曾经在项目中使用过的一些内容(当然,未必所有技巧的使用频率都很高_)。
本篇文章主要是探讨一些辅助函数的书写优化。
函数优化
在JavaScript中由于浏览器的兼容性差异,导致了在操纵DOM相关的代码中常常会有所差异,比如最经典的给DOM绑定事件
//dom
var dom = document.getElementById('example');
//event
function clickEvent () {
console.log('click!');
}
//IE9以上及Chrome、Firefox
dom.addEventListener('click', clickEvent, false);
//IE
dom.attachEvent('onclick', clickEvent);
//全部通用
dom.onclick = clickEvent;
可以看出给DOM绑定事件可以有三种实现方式,而我们会依次采用addEventListener/attachEvent/onclick这样的排序去绑定事件。聪明的我们可能就会立即想到我们应该要用一个通用函数来处理,毕竟谁也不想每次绑定事件都要判断一次浏览器特性。
function BindEvent (dom, event, handle, ex) {
if ('addEventListener' in dom) {
dom.addEventListener(event, handle, ex || false);
} else if ('attachEvent' in dom) {
dom.attachEvent('on' + event, handle);
} else {
dom['on' + event] = handle;
}
}
这看起来非常棒,使用起来也不错。
var dom = document.getElementById('example');
BindEvent(dom, 'click', function () {
alert('hey!');
});
现实情况往往会比想象中复杂那么一点点,页面上不可能只有一两个dom,在一些SPA中甚至会成千上万,来让我们尝试一下优化。其实对于浏览器的特性侦测只需要执行一次即可,而没有必要每次绑定事件的时候都去判断,我们不妨将BindEvent改成立即执行函数。
var BindEvent = (function () {
if ('addEventListener' in document) {
return function (dom, event, handle, ex) {
dom.addEventListener(event, handle, ex || false);
}
} else if ('attachEvent' in document) {
return function (dom, event, handle) {
dom.attachEvent('on' + event, handle);
}
} else {
return function (dom, event, handle) {
dom['on' + event] = handle;
}
}
})();
这样BindEvent就会在加载的时候直接采用了最符合当前浏览器情况的实现方式。最后让我们来写一些测试代码对比一下效率,对于我们页面上最可能出现的情况(有1000个DOM)。这里我们采用Chrome提供的console.profile函数来做性能分析
function GenerateDOM (start, end) {
for (; start < end; start++) {
var dom = document.createElement('div');
dom.id = 'd' + start;
dom.innerHTML = 'I\'m d' + start + ' !';
document.body.appendChild(dom);
}
}
GenerateDOM(0, 1000);
function BindDOMEvent (doms, event, handle, ex) {
for (var i = 0; i < doms.length; i++) {
BindEvent(doms[i], event, handle, ex);
}
}
var doms = document.getElementsByTagName('div');
console.profile();
BindDOMEvent(doms, 'click', function () {
alert('hey! ' + this.id);
});
console.profileEnd();
当你在页面按F12打开console点击Profiles的tab你可以看到Chrome提供的分析结果。而我们暂时只需要分析BindDOMEvent的耗时即可。如下图:

以下是我分别各执行两种版本10次后得出的综合分析结果,可以看出避免重复判断后,性能平均快了0.7ms,而且优化后基本能够维持在2.1ms(快了约1.27倍)
原始版本
3.3 / 3.4 / 3.3 / 2.3 / 3.5 / 2.3 / 3.4 / 3.3 / 2.2 / 3.3
avg: 3.03 max: 3.5 min: 2.2
优化后版本
2.1 / 2.1 / 2.1 / 3.4 / 2.1 / 2.2 / 2.2 / 3.2 / 2.2 / 2.1
avg: 2.37 max: 3.4 min: 2.1
所以,我们可以直观的判断出,当有类似重复判断的函数,不妨观察分析一下是否能够将该判断只进行一次。这样已经能够给我们的性能带来一定的提升。
优化之路永无止境
对于大多数的项目来说,类似BindEvent这样的函数一般都会封装成一个Library以便共用代码。而这样的公用Library常常是每个页面都会默认加载。
<script src="/public/helper.js"></script>
但是实际上每个页面所用到的功能仅仅是Library里面的一小部分,例如我们刚刚所写的BindEvent。我们回头看一看实现代码。
var BindEvent = (function () {
if ('addEventListener' in document) {
return function (dom, event, handle, ex) {
dom.addEventListener(event, handle, ex || false);
}
} else if ('attachEvent' in document) {
return function (dom, event, handle) {
dom.attachEvent('on' + event, handle);
}
} else {
return function (dom, event, handle) {
dom['on' + event] = handle;
}
}
})();
这里有什么可以优化的点呢?在于我们不确定这个页面是否会用到这个function的时候,我们都默认帮他初始化好了,这里面会有一定的性能损耗。毕竟一个Library当中这样的方法会有N个,当N个函数都在页面加载的时候进行初始化,对于整个页面加载速度方面的影响不容小瞧。
那怎么去做优化呢?惰性加载,只有当有调用方去触发这个function的时候我们才去做初始化。实现代码如下:
var BindEvent = function (dom, event, handle, ex) {
if ('addEventListener' in document) {
BindEvent = function (dom, event, handle, ex) {
dom.addEventListener(event, handle, ex || false);
}
} else if ('attachEvent' in document) {
BindEvent = function (dom, event, handle) {
dom.attachEvent('on' + event, handle);
}
} else {
BindEvent = function (dom, event, handle) {
dom['on' + event] = handle;
}
}
BindEvent(dom, event, handle, ex);
};
这样当BindEvent被调用了一次之后,我们的BindEvent函数就会被替换成符合浏览器特性的实现方式,之后的所有调用均是调用最优结果。最后我们再进行同样地测试看一下对比
3.2 / 2.1 / 2.2 / 3.2 / 3.2 / 2.1 / 3.1 / 2.1 / 3.2 / 3.2
avg: 2.76 max: 3.2 min: 2.1
可以看到平均耗时是2.75而之前是2.37,性能仅仅是略为减慢而已(约是1.16倍),影响并不是非常大,可是对于页面加载的影响呢?
来,让我们用console.time来做一下对比测试,看一下两种实现方式对于页面加载的影响。
console.time('event1');
var BindEvent = (function () {
if ('addEventListener' in document) {
return function (dom, event, handle, ex) {
dom.addEventListener(event, handle, ex || false);
}
} else if ('attachEvent' in document) {
return function (dom, event, handle) {
dom.attachEvent('on' + event, handle);
}
} else {
return function (dom, event, handle) {
dom['on' + event] = handle;
}
}
})();
console.timeEnd('event1');
console.time('event2');
var BindEvent2 = function (dom, event, handle, ex) {
if ('addEventListener' in document) {
BindEvent = function (dom, event, handle, ex) {
dom.addEventListener(event, handle, ex || false);
}
} else if ('attachEvent' in document) {
BindEvent = function (dom, event, handle) {
dom.attachEvent('on' + event, handle);
}
} else {
BindEvent = function (dom, event, handle) {
dom['on' + event] = handle;
}
}
BindEvent(dom, event, handle, ex);
};
console.timeEnd('event2');
测试结果中前面的运行毫秒是BindEvent而后面的则是BindEvent2
0.038 / 0.007
0.041 / 0.006
0.031 / 0.007
0.053 / 0.009
0.041 / 0.007
BindEvent平均耗时: 0.0408 而 BindEvent2平均耗时: 0.00702
前者约是后者的5.81倍
通过整体评测可以看得出来对于一个能够采用惰性加载的函数,通过这样的一个方式,能够大大减少页面加载时间,同时不会对执行时的运行效率产生太大的影响。
结论
- 对于需要重复判断的条件看能否提取出来只判断一次
- 如果不能,也可以选择保存一个
bool值变量来做,而不要进行过于复杂的判断
- 如果不能,也可以选择保存一个
- 对于不是页面必须的函数,可以采用惰性加载方式来实现
怎么样?你学会了这个技巧了么?
console.profile与console.time的文档 需要自备梯子- -!
JavaScript的妙与乐(一)之 函数优化的更多相关文章
- 我自己的Javascript 库,封装了一些常用函数 Kingwell.js
我自己的Javascript 库,封装了一些常用函数 Kingwell.js 博客分类: Javascript javascript 库javascript库 现在Javascript库海量,流行的 ...
- 博文推荐】Javascript中bind、call、apply函数用法
[博文推荐]Javascript中bind.call.apply函数用法 2015-03-02 09:22 菜鸟浮出水 51CTO博客 字号:T | T 最近一直在用 js 写游戏服务器,我也接触 j ...
- JavaScript字符串插入、删除、替换函数
JavaScript字符串插入.删除.替换函数 说明: 以下函数中前两个函数取出查找字符串的前一部分和后一部分,以用于其他函数.注意,调用一次 replaceString(mainStr,search ...
- 使用 JavaScript 实现名为 flatten(input) 的函数,可以将传入的 input 对象(Object 或者 Array)进行扁平化处理并返回结果
请使用 JavaScript 实现名为 flatten(input) 的函数,可以将传入的 input 对象(Object 或者 Array)进行扁平化处理并返回结果.具体效果如下: const in ...
- 用JavaScript写一个类似PHP print_r的函数
PHP print_r的函数很好用,可以用来打印数组.对象等的结构与数据,可惜JavaScript并没有原生提供类似的函数.不过我们可以试着自己来实现这个函数,下面提供一些方法与思路. 方法一 fun ...
- javascript进阶课程--第三章--匿名函数和闭包
javascript进阶课程--第三章--匿名函数和闭包 一.总结 二.学习要点 掌握匿名函数和闭包的应用 三.匿名函数和闭包 匿名函数 没有函数名字的函数 单独的匿名函数是无法运行和调用的 可以把匿 ...
- JavaScript深入浅出第1课:箭头函数中的this究竟是什么鬼?
<JavaScript 深入浅出>系列: JavaScript 深入浅出第 1 课:箭头函数中的 this 究竟是什么鬼? JavaScript 深入浅出第 2 课:函数是一等公民是什么意 ...
- javascript中,一个js中的函数,第一句var _this = this;为什么要这样做?
javascript中,一个js中的函数,第一句var _this = this;为什么要这样做? 下面是源码: 下面这段代码是常用的网站首页,自动切换span或者tabbar来变更List显示内容的 ...
- JavaScript学习总结(四)function函数部分
转自:http://segmentfault.com/a/1190000000660786 概念 函数是由事件驱动的或者当它被调用时执行的可重复使用的代码块. js 支持两种函数:一类是语言内部的函数 ...
随机推荐
- 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新
本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...
- nohup程序后台执行
Linux常用命令,用于不挂断的执行程序. nohup命令:如果你正在运行一个进程,而且你觉得在退出帐户时该进程还不会结束,那么可以使用nohup命令.该命令可以在你退出帐户/关闭终端之后继续运行相应 ...
- 拨开迷雾,找回自我:DDD 应对具体业务场景,Domain Model 到底如何设计?
写在前面 除了博文内容之外,和 netfocus 兄的讨论,也可以让你学到很多(至少我是这样),不要错过哦. 阅读目录: 迷雾森林 找回自我 开源地址 后记 毫无疑问,领域驱动设计的核心是领域模型,领 ...
- Partition:分区切换(Switch)
在SQL Server中,对超级大表做数据归档,使用select和delete命令是十分耗费CPU时间和Disk空间的,SQL Server必须记录相应数量的事务日志,而使用switch操作归档分区表 ...
- Bootstrap 模态框(Modal)插件
页面效果: html+js: <!DOCTYPE html> <html> <head> <meta charset="utf-8"> ...
- JavaScript之链式结构序列化
一.概述 在JavaScript中,链式模式代码,太多太多,如下: if_else: if(...){ //TODO }else if(...){ //TODO }else{ //TODO } swi ...
- 谈谈一些有趣的CSS题目(五)-- 单行居中,两行居左,超过两行省略
开本系列,讨论一些有趣的 CSS 题目,抛开实用性而言,一些题目为了拓宽一下解决问题的思路,此外,涉及一些容易忽视的 CSS 细节. 解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题中有你感觉 ...
- ASP.NET从零开始学习EF的增删改查
ASP.NET从零开始学习EF的增删改查 最近辞职了,但是离真正的离职还有一段时间,趁着这段空档期,总想着写些东西,想来想去,也不是很明确到底想写个啥,但是闲着也是够 ...
- MFC单文档程序添加HTML帮助支持
1.在App类 构造函数中添加 EnableHtmlHelp(); 2.在Frame类中,添加消息影射: ON_COMMAND(ID_HELP_FINDER, CFrameWnd::OnHelpFin ...
- MVC还是MVVM?或许VMVC更适合WinForm客户端
最近开始重构一个稍嫌古老的C/S项目,原先采用的技术栈是『WinForm』+『WCF』+『EF』.相对于现在铺天盖地的B/S架构来说,看上去似乎和Win95一样古老,很多新入行的,可能就没有见过经典的 ...