参考高性能javascript

理解浏览器UI线程  用于执行javascript和更新用户界面的进程通常被称为浏览器UI线程  UI线程的工作机制可以理解为一个简单的队列系统,队列中的任务按顺序执行

<button onclick="handleClick()">click</button>
<script type="text/javascript">
function handleClick() {
var div = document.createElement('div');
div.innerHTML = "aaa";
document.body.appendChild(div);
}

在上面的例子中当按钮被点击的时候,它触发UI线程创建的两个任务并且添加到队列中,更新被点击按钮的UI和执行javascript代码(handleClick) 假如此刻队列是空闲的(也就数存在用户操作无法添加到队列的情况) 第一个任务会立即被执行然后javascript代码被提取出来并且执行,在执行javascript代码的过程中创建了一个div,当js运行完,又触发了另一次的UI更新(显示创建的DIV)

由于js脚本运行时间的长短会导致新创建的任务无法被加入队列,所以有必要对脚本的执行时间进行限制或者在脚本执行的过程中让出一会浏览器的UI线程让新的任务添加到队列

理解定时器  参考我之前的一篇blog  定时器相关setTimeout setInterval requestAnimationFrame

通过上面的一些点 定时器可以称为长时间运行脚本的跨浏览器的解决方案

使用定时器取代循环

  • 处理过程是否必须同步
  • 数据是否一定要按顺序处理

如果上面的两个回答都是否,那我们就可以使用定时器来代替循环了

  function processArray(items,process,callback) {
var todo = items.concat();
setTimeout(function(){
process(todo.shift()); if(todo.length > 0) {
setTimeout(arguments.callee,25);
} else {
callback(items);
}
},25);
}

上面的函数接受一个数组项  处理数组项的函数  回调函数    通过自调用的方式能保证队列中不会有缺失的代码片段也保证了代码执行的间隔至少是25ms

上面方式的缺陷是延长了处理数组的总时长(加入了时间间隔) 但是给出了一定时间间隔留给了UI线程,能避免代码长时间运行导致浏览器锁死

改进版的定时器取代循环

  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();
}
}); }

分割任务

  将一个任务划分成多个原子任务,使用定时器去执行相应的原子任务,思路跟上面的循环处理数组的思路相同

  function multistep(steps,args,callback) {
var tasks = steps.concat(); setTimeout(function(){
var task = tasks.shift();
task.apply(null,args || []);
if(tasks.length > 0) {
setTimeout(arguments.callee,25)
} else {
callback();
}
},25)
} //这里的args必须是数组 因为我们使用的apply

了解web workers  通过定义一个包含worker代码的文件 通过new Worker("js文件url")就能在网页中创建一个新的线程,它通过特定的方法与主线程进行信息的交互,并且不影响主线程的执行

主线程
var worker = new Worker("worker.js"); //这段代码会异步的下载worker.js 当worker.js下载并执行完成后就启动了这个worker
worker.onmessage = function(event) {
alert(event.data);
} //网页代码接收数据的处理函数
worker.postMessage("hello");//网页代码向worker发送数据 worker.js self.onmessage = function(event) {
self.postMessage(event.data+"world");
} //worker接收网页代码数据的处理函数

只有特定的数据可以通过postMessage传递 例如原始值(字符串 数字 布尔值 null undefined ) 也可以传递Object和Array实例 数据会被序列化出入和传出worker 然后被反序列化

用处: web workers适用于处理纯数据或者与浏览器UI无关的长时间运行的脚本

编程实践

1.避免双重求值

eval()   Function()   setTimeout() setInterval() 允许在程序中提取一个包含代码的字符串,然后动态的的执行它  例如eval("num1 + num2") 这段代码会先对eval进行求值 然后对字符串中的内容进行求值返回结果  应该避免使用eval 和Function()  并且在使用setTimeout  和 setInterval的时候传入函数体

2. 使用object 和 array 直接量

3 避免重复的工作   例如跨浏览器的事件绑定函数

function addListenEvent(eventTarget,eventType,handler) {
if(eventTarget.addEventListener) {
eventTarget.addEventListener(eventType,handler,false);
} else if(eventTarget.attachEvent) {
eventType = "on" + eventType;
eventTarget.attachEvent(eventType,handler);
} else {
eventTarget["on"+eventType] = handler;
}
}

这个方法在每次调用的时候都进行了浏览器的检测,有重复的操作

我们可以通过延时加载和条件预加载的方式去优化这个函数

1延时加载 函数在使用之前不进行任何的操作,在调用后在决定根据情况去改变这个函数的状态 这种方式在第一次运行的时候会消耗较长的时间,适合当一个函数不会被立即调用的情况下使用  使用延时加载改写上面的函数

function addListenEvent(eventTarget,eventType,handler) {
if(eventTarget.addEventListener) {
addListenEvent = function(eventTarget,eventType,handler) {
eventTarget.addEventListener(eventType,handler,false);
}
} else if(eventTarget.attachEvent) {
addListenEvent = function(eventTarget,eventType,handler) {
eventType = "on" + eventType;
eventTarget.attachEvent(eventType,handler);
}
} else {
addListenEvent = function(eventTarget,eventType,handler) {
eventTarget["on"+eventType] = handler;
}
}
}

2条件预加载     这种方式需要在脚本加载的时候就进行检测,适用于一个函数马上就需要被调用并且在整个页面的生命周期被频繁使用的场景

var addListenEvent = document.body.addEventListener ? function(eventTarget,eventType,handler) {
eventTarget.addEventListener(eventType,handler,false);
} : function(eventTarget,eventType,handler) {
eventType = "on" + eventType;
eventTarget.attachEvent(eventType,handler);
};

使用速度快的运算方式

例如在判断奇数偶数的时候 通过与1进行按位与的来代替取余操作能提高性能

使用原生的方法

小记  ~25 = -26

25的二进制形式为   11001  取反的二进制形式为 00110   左边   右边  因为~x+1 = -x 所以右边~26 + 1 = 00110

高性能javascript学习笔记系列(5) -快速响应的用户界面和编程实践的更多相关文章

  1. 高性能javascript学习笔记系列(2)-数据存取

    参考 高性能javascript Tom大叔深入理解javascript系列 相关概念 1.执行上下文   当控制器转到ecmascript可执行代码的时候,就会进入一个执行上下文,执行上下文是以堆栈 ...

  2. 高性能javascript学习笔记系列(1) -js的加载和执行

    这篇笔记的内容主要涉及js的脚本位置,如何加载js脚本和脚本文件执行的问题,按照自己的理解结合高性能JavaScript整理出来的 javascript是解释性代码,解释性代码需要经历转化成计算机指令 ...

  3. 高性能javascript学习笔记系列(6) -ajax

    参考 高性能javascript javascript高级程序设计 ajax基础  ajax技术的核心是XMLHttpRequest对象(XHR),通过XHR我们就可以实现无需刷新页面就能从服务器端读 ...

  4. 高性能javascript学习笔记系列(4) -算法和流程控制

    参考高性能javascript for in 循环  使用它可以遍历对象的属性名,但是每次的操作都会搜索实例或者原型的属性 导致使用for in 进行遍历会产生更多的开销 书中提到不要使用for in ...

  5. 高性能javascript学习笔记系列(3) -DOM编程

    参考 高性能javascript 文档对象模型(DOM)是独立于语言的,用于操作XML和HTML文档的程序接口API,在浏览器中主要通过DOM提供的API与HTML进行交互,浏览器通常会把DOM和ja ...

  6. JavaScript学习笔记系列2:Dom操作(一)

    一.什么是Dom? DOM------>Document Object Model 直接翻译就是文档对象模型. DOM------>定义了表示和修改文档所需的对象.这些对象的行为和属性以及 ...

  7. 高性能javascript 学习笔记(1)

    加载和运行 管理浏览器中的javascript代码是个棘手的问题,因为代码运行阻塞了其他浏览器处理过程,诸如用户绘制,每次遇到<script>标签,页面必须停下来等待代码下载(如果是外部的 ...

  8. JavaScript学习笔记系列1:JavaScript的是什么?

    一.JS是什么? JavaScript作为Netscape Navigator浏览器的一部分首次出现在1996年.它最初的设计目标是改善网页的用户体验. 作者:Brendan Eich 期初JavaS ...

  9. Dynamic CRM 2013学习笔记 系列汇总

    这里列出所有 Dynamic CRM 2013学习笔记 系列文章,方便大家查阅.有任何建议.意见.需要,欢迎大家提交评论一起讨论. 本文原文地址: Dynamic CRM 2013学习笔记 系列汇总 ...

随机推荐

  1. WCF学习之旅——第一个WCF示例(二)

    第四步:通过自我寄宿的方式寄宿服务 WCF服务需要依存一个运行着的进程(宿主),服务寄宿就是为服务指定一个宿主的过程.WCF是一个基于消息的通信框架,采用基于终结点(Endpoint)的通信手段. 终 ...

  2. Ajax_01之概述、响应

    1.URL.URI和URN URL:Unified Resource Locator:统一资源定位符: URI:Unified Resource Identifier:统一资源识别符: URN:Uni ...

  3. android studio上的基本动画实现(第一篇)

    hello,各位小伙伴们,在很多小伙伴们刚刚开始学习android的时候,常常会有一些project里面需要有一些基本动画的插入,那么具体是要怎么实现呢?我们接下一起分析一下在android中的几种基 ...

  4. OleDB Destination 用法

    第一部分:简介 OleDB Destination component 是将数据流load 到destination,共有5种Data Access Mode,一般的Destination compo ...

  5. Ado net Source 用法

    Ado net Source 是用于获取数据源的,使用的connection manager是 ado net connection. Ado Net Source 的Data Access Mode ...

  6. ScriptTask读取SharePoint上的Excel

    很多情况下,我们会SharePoint 作为文件共享的Server,有时,业务上需要将SharePoint上的Excel文档 Download 到db中.对于搞DB的人来说,难点是如何下载Excel? ...

  7. [WPF]带下拉列表的文本框

    控件我已经弄好了,代码比较多,所以没办法全面介绍. 一开始我是直接继承Selector类来实现,做是做出来了,不过发现性能不太好.于是,我就想着自己来实现.毕竟我是做给自己用的,也不考虑过多的东西,也 ...

  8. Android之SqlLite数据库使用

    每个应用程序都要使用数据,Android应用程序也不例外,Android使用开源的.与操作系统无关的SQL数据库—SQLite.SQLite第一个Alpha版本诞生于2000年5月,它是一款轻量级数据 ...

  9. gsoap设置超时

    1.修改gsoap自动生成的代码才能进行超时设置(我这边访问web service的代码都是gsoap工具自动生成.根据wsdl接口) 2.找到生成的soapwwwsdlBindingProxy.cp ...

  10. Java 8新特性-5 内建函数式接口

    在之前的一片博文 Lambda 表达式,提到过Java 8提供的函数式接口.在此文中,将介绍一下Java 8四个最基本的函数式接口 对于方法的引用,严格来讲都需要定义一个接口.不管我们如何操作实际上有 ...