JavaScript性能优化汇总

JavaScript性能优化汇总

本文主要针对《高性能JavaScript》书中提到的优化方案进行总结,只列出文中推荐最高效的方案,其他方案的信息请查看之前的文章。

《高性能JavaScript》第一章 加载和执行
《高性能JavaScript》第二章 数据存取
《高性能JavaScript》第三章 DOM编程
《高性能JavaScript》第四章 算法和流程控制
《高性能JavaScript》第五章 字符串和正则表达式
《高性能JavaScript》第六章 快速响应的用户界面
《高性能JavaScript》第七章 Ajax
《高性能JavaScript》第八章 编程实践
《高性能JavaScript》第九章 构建并部署高性能JavaScript应用
《高性能JavaScript》第十章 工具

第一章 加载和执行

优化原则

  1. 脚本放在<body>标签底部加载,即</body>前面;
  2. 将多个脚本合并到一个脚本中引用,避免一个页面引用多个脚本;
  3. <script>标签添加defer属性延迟加载脚本;
  4. 根据需求,在合适的时间动态加载各个脚本;

代码示例

先添加动态加载所需的代码,然后加载初始化页面所需的剩下的代码。

<script type="text/javascript">
    function loadScript(url, callback){
        var script = document.createElement("script")
        script.type = "text/javascript";
        if (script.readyState){ // IE
            script.onreadystatechange = function(){
                if (script.readyState == "loaded" || script.readyState == "complete"){
                    script.onreadystatechange = null;
                    callback();
                }
            };
        } else { // Others
            script.onload = function(){
                callback();
            };
        }
        script.src = url;
        document.getElementsByTagName("head")[0].appendChild(script);
    }
    loadScript("the-rest.js", function(){
        Application.init();
    });
</script>

第二章 数据存取

优化原则

  1. 若跨作用域的值引用一次以上,请将其存入局部变量中重复使用;
  2. 避免使用with语句,谨慎使用try-catch语句,因为它们会改变作用域链;
  3. 除非确实有必要,才去使用动态作用域;
  4. 小心使用闭包,防止内存泄漏;
  5. 对象成员嵌套越深,读取速度就越慢;
  6. 将对象成员、数组元素像跨域变量一样存入局部变量来使用,同时避免用在对象的成员方法上,因为this的存在。

代码示例

// 引用一次document
function initUI(){
    var doc = document,
    bd = doc.body,
    links = doc.getElementsByTagName("a"), 
    i = 0,
    len = links.length;
    while(i < len){
        update(links[i++]);
    }
    doc.getElementById("go-btn").onclick = function(){
        start();
    };
    bd.className = "active";
}

第三章 DOM编程

优化原则

  1. 减少DOM访问次数,将运算尽量留在JavaScript端来处理;
  2. 在对性能要求苛刻的情况下,推荐使用innerHTML而非DOM原生方法更新一大段HTML;
  3. 使用cloneNode代替createElement来更新页面,性能稍微好点;
  4. 将HTML集合拷贝到数组中,访问它的属性会更快;
  5. 使用能区分元素节点的API获取目标节点,避免取出其他类型的节点;
  6. 利用CSS选择器,使用querySelectorAllquerySelector来获取节点效率更快;
  7. 尽量少的改变元素的几何属性,防止页面重排或重绘;
  8. 避免使用能刷新渲染队列的属性,如offsetTop等;
  9. 合并多次对DOM和样式的修改,然后一次处理掉;
  10. 大量数据下,避免使用:hover这个CSS伪选择器;
  11. 使用事件委托处理DOM事件,减少事件处理器的数量。

代码示例

function collectionNodesLocal() {
    var coll = document.getElementsByTagName('div'),
    len = coll.length,
    name = '',
    el = null;
    for (var count = 0; count < len; count++) {
        el = coll[count];
        name = el.nodeName;
        name = el.nodeType;
        name = el.tagName;
    }
    return name;
};

第四章 算法和流程控制

优化原则

  1. 避免使用for-in循环,使用其他循环代替它;
  2. 最小化循环中的属性查找,并使用倒序循环;
  3. 使用达夫设备模式减少迭代次数;
  4. 在运行速度要求严格时,避免使用forEach循环;
  5. 根据离散值和值域,以及条件判断次数,选择使用if-elseswitch
  6. 使用查找表代替条件语句,即将返回值集合存入数组中,操作数据获取对应的值;
  7. 递归可能会导致浏览器的调用栈大小限制问题,将递归改为迭代可以避免;
  8. 使用Memoization缓存计算结果,避免重复计算。

代码示例

//最小化属性查找并反转:比原始版本快50%~60%
for (var i=items.length; i--; ){
    process(items[i]);
}
var j = items.length;
while (j--){
    process(items[j]]);
}
var k = items.length-1;
do {
    process(items[k]);
} while (k--);
// 升级版Duff’s Device:尽管这种方式用两次循环代替之前一次循环,但移除了循环体中的switch语句,速度比原始更快。
var i = items.length % 8;
while(i){
    process(items[i--]);
}
i = Math.floor(items.length / 8);
while(i){
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
    process(items[i--]);
}
// 将返回值存入数组
var results = [result0, result1, result2, result3, result4, result5, result6, result7, result8, result9, result10]
// 利用数组返回当前结果
return results[value];

第五章 字符串和正则表达式

优化原则

  1. 以原有字符串作为基础合并字符串,避免使用临时字符串;
  2. 避免使用Array.prototype.join合并数组中的字符串;
  3. 避免使用String.prototype.concat合并字符串;
  4. 尽可能具体正则表达式中化分隔符之间的字符串匹配模式;
  5. 使相邻的字元互斥,避免嵌套量词对同一字符串的相同部分多次匹配,通过重复利用预查的原子组去除不必要的回溯。

代码示例

// 以下优化会提速10%~40%,原因是这样做避免使用了临时字符串。
str += "one";
str += "two";
等价于
str = str + "one" + "two";
等价于
str = ((str + "one") + "two")

第六章 快速响应的用户界面

优化原则

  1. 源于Robert Moller 1968年的研究,界面反应时间超过100毫秒,用户就会感到自己与界面失去联系;
  2. 使用定时器释放UI控制权,定时器延迟至少25毫秒,保证UI更新;
  3. 使用定时器处理数组循环,增加了处理总耗时,但避免了UI锁定带来糟糕的用户体验;
  4. 将耗时较长的任务分割成多个小任务,避免不好的体验;
  5. 一个定时器可以同时处理多个任务,但需保证一轮总耗时不超过50毫秒
  6. 使用Web Workers在UI线程外执行JavaScript代码。

代码示例

// 一个定时器处理多个小任务
function timedProcessArray(items, process, callback){
    var todo = items.concat(); // 克隆原始数组
    setTimeout(function(){
        var start = +new Date(); // +可以把Date转换成数字,方便计算
        do {
            process(todo.shift());
        } while (todo.length > 0 && (+new Date() - start < 50));
        if (todo.length > 0){
            setTimeout(arguments.callee, 25);
        } else {
            callback(items);
        } 
    }, 25);
}

第七章 Ajax

优化原则

  1. 使用JSON或JSON-P代替其他数据结构;
  2. 自定义数据格式非常简洁,下载和解析都很快,但不易阅读;
  3. 使用Expires头信息将数据缓存在浏览器;
// 简化后的JSON数据
[
    [ 1, "alice", "Alice Smith", "alice@alicesmith.com" ],
    [ 2, "bob", "Bob Jones", "bob@bobjones.com" ],
    [ 3, "carol", "Carol Williams", "carol@carolwilliams.com" ],
    [ 4, "dave", "Dave Johnson", "dave@davejohnson.com" ]
]
// 自定义数据结构
1:alice:Alice Smith:alice@alicesmith.com;
2:bob:Bob Jones:bob@bobjones.com; 
3:carol:Carol Williams:carol@carolwilliams.com;
4:dave:Dave Johnson:dave@davejohnson.com

第八章 编程实践

优化原则

  1. 避免使用eval()Function(),小心使用setTimeout()setInterval(),因为可能会造成双重求值;
  2. 使用数组和对象直接量,避免使用new
  3. 使用延迟加载或条件预加载来避免重复工作;
  4. 优先使用位操作和原生方法来提高性能。

代码示例

var myObject = { // 创建对象
    name: "Nicholas",
    count: 50,
    flag: true,
    pointer: null
};
var myArray = ["Nicholas", 50, true, null]; // 创建数组
// 延迟加载
function addHandler(target, eventType, handler){
    // 复写现有函数
    if (target.addEventListener){ //DOM2 Events
        addHandler = function(target, eventType, handler){
            target.addEventListener(eventType, handler, false);
        };
    } else { //IE
        addHandler = function(target, eventType, handler){
            target.attachEvent("on" + eventType, handler);
        };
    }
    // 调用新函数 
    addHandler(target, eventType, handler);
}
// 条件预加载
var addHandler =
    document.body.addEventListener
    ?
    function(target, eventType, handler){
    target.addEventListener(eventType, handler, false);}
    :
    function(target, eventType, handler){
    target.attachEvent("on" + eventType, handler);};

第九章 构建并部署高性能JavaScript应用

优化原则

  1. 使用构建工具自动化构建工程;
  2. 使用工具合并多个JavaScript文件;
  3. 使用工具预处理JavaScript文件;
  4. 使用工具压缩JavaScript;
  5. Accept-Encoding头进行HTTP压缩;
  6. 使用内容分发网络CDN提升响应速度。

第十章 工具

主要介绍常见的性能分析和网络分析工具,不过由于时间关系,这些工具大多数已经过时了,在此就不多做介绍了。

欢迎大佬纠错指导,欢迎同行交流学习。

GitHub:https://github.com/Code4GL
简书:https://www.jianshu.com/u/7f5541a6b6d2
知乎:https://www.zhihu.com/people/code4gl/activities
公众号:code_everything
QQ:771841496
邮箱:guanli1991@163.com

扫码关注公众号code_everything
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,527评论 5 470
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,314评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,535评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,006评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,961评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,220评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,664评论 3 392
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,351评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,481评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,397评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,443评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,123评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,713评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,801评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,010评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,494评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,075评论 2 341

推荐阅读更多精彩内容