再谈JavaScript线程


Posted in Javascript onJuly 10, 2015

代码判断一:

<div id="div">
  click me
</div>

<script>
  var div=document.getElementById("div");
  div.addEventListener('click',function(){
   alert('You have clicked me!');
  });
  for(var i =0; i<999999999;i++){
   console.log(i);
  }
</script>

执行之,不出意外的话所有浏览器都会卡死,因为上面的for循环次数太多了,非常耗费CPU资源,而基于JavaScript单线程的事实,浏览器UI渲染被挂起而导致假死。

       现在问题来了,我就是想要实现上述代码,怎么办?

Concurrent.Thread.js
       该类库实质上还是使用setTimeout来实现一个“假的多线程”。在HTML5 WebWorker问世之前是一个很好的选择。比如我们要实现上述“代码片段一”,可以这样写(点我下载类库):

代码片段二:

<div id="div">
  click me
</div>
<script src="Concurrent.Thread.js"></script>
<script>
  Concurrent.Thread.create(function(){
    var div=document.getElementById("div");
    div.addEventListener('click',function(){
     alert('You have clicked me!');
    });
    for(var i =0; i<9999999;i++){
     console.log(i);
    }
  });
</script>

 通过该类库提供的create方法可以创建一个“新线程”。另外,给script标签的type属性设置为 text/x-script.multithreaded-js 也可以实现同样的效果:

代码片段三:

<div id="div">
  click me
</div>
<script src="Concurrent.Thread.js"></script>
<script type="text/x-script.multithreaded-js">
    var div=document.getElementById("div");
    div.addEventListener('click',function(){
     alert('You have clicked me!');
    });
    for(var i =0; i<9999999;i++){
     console.log(i);
    }
</script>

WebWorker
    针对以上浏览器卡死这种糟糕的用户体验,HTML5怎么会熟视无睹呢?

     下面我们用经典的斐波那契数列来做测试:

代码片段四:

主页面:

<div id="div"></div>
<script>
  window.onload=function(){
     var div=document.getElementById("div");
     if(typeof(Worker)!=="undefined"){//在创建WebWorker之前,先判断浏览器是否支持


  console.log("Start calculating....");
       var time1= new Date()*1;//获得当前时间戳
       var worker=new Worker("fibonacci.js");//创建WebWorker对象,并传递在新线程中将要执行的脚本的路径
       worker.onmessage=function(e){ //监听从新线程发送过来的数据
         div.innerHTML=e.data;
         var time2=new Date()*1;
         console.log("time spend:"+(time2-time1)+"ms");
       }





       worker.postMessage(36);//向新线程发送数据
     }else{
       alert("Your browser do not support WebWoker");
     }
  }
</script>

fibonacci.js:
var fibonacci=function (n){
  return n<3?n:(arguments.callee(n-1)+arguments.callee(n-2));
}
onmessage=function(e){
  var num=parseInt(e.data,10);
  postMessage(fibonacci(num));//向主页面发送数据
}

基本的使用方法已在代码中做注释了,查看控制台,可以看见很快就打印出执行时间了。所以我们得出的结论是:WebWorker适合在前端执行复杂的大量的计算。需要注意的是,WebWorker不支持跨域,本地测试还是用http协议,不要用file协议,否则不能创建Worker对象而报脚本错误 。

再谈JavaScript线程

        如果我们需要连续执行多个postMessage操作,最好不要work.postMessage一直写,像这样:

worker.postMessage(36);

    worker.postMessage(36);

    worker.postMessage(36);

       因为此时只有一个WebWorker实例,postMessage会顺序执行而不是异步执行,就不能充分发挥它的性能了。可以通过创建多个WebWorker实例来发送数据。

        需要注意的几点事项有:

        1、我们观察到WebWorker通过接受一个url来创建一个worker,而jsonp的实现原理就是通过动态插入script标签加载数据,那我们尝试用WebWorker来实现同样的事情不是更好吗?因为WebWorker是多线程的,没有阻塞,岂不美哉?但实际上经过实验,我们发现WebWorker表现并不如意。所以这并不是它擅长的事,我们还是不要让它越俎代庖的好。

        2、WebWorker在接受其他来源信息的时候,其实也给站点的安全带来了隐患,如果接收不明来源的脚本信息,可能会导致XSS注入攻击。所以这点需要防范,其实我们上面例子中使用innerHTML是不安全的,可以使用innerText或现代浏览器提供的textContent来替代,以过滤掉html标签。

今天比较累了,想睡觉了,先写这么多吧。

Javascript 相关文章推荐
javascript 操作Word和Excel的实现代码
Oct 26 Javascript
extjs两个tbar问题探讨
Aug 08 Javascript
JS将秒换成时分秒实现代码
Sep 03 Javascript
js使用栈来实现10进制转8进制与取除数及余数
Jun 11 Javascript
js 数组去重的四种实用方法
Sep 09 Javascript
基于javascript实现窗口抖动效果
Jan 03 Javascript
JavaScript添加随滚动条滚动窗体的方法
Feb 23 Javascript
jQuery视差滚动效果网页实现方法经验总结
Sep 29 Javascript
基于原生js淡入淡出函数封装(兼容IE)
Oct 20 Javascript
Webpack中雪碧图插件使用详解
May 25 Javascript
JavaScript实现猜数字游戏
May 20 Javascript
详解vue实现坐标拾取器功能示例
Nov 18 Vue.js
javascript实现判断鼠标的状态
Jul 10 #Javascript
javascript检查浏览器是否已经启用XX功能
Jul 10 #Javascript
javascript实现确定和取消提示框效果
Jul 10 #Javascript
javascript实现的简单的表单验证
Jul 10 #Javascript
jQuery原生的动画效果
Jul 10 #Javascript
简述Jquery与DOM对象
Jul 10 #Javascript
JavaScript声明变量名的语法规则
Jul 10 #Javascript
You might like
用sql命令修改数据表中的一个字段为非空(not null)的语句
2010/06/04 PHP
php面向对象中的魔术方法中文说明
2014/03/04 PHP
PHP中上传多个文件的表单设计例子
2014/11/19 PHP
PHP 绘制网站登录首页图片验证码
2016/04/12 PHP
PHP读取zip文件的方法示例
2016/11/17 PHP
JS重要知识点小结
2011/11/06 Javascript
js运动框架_包括图片的淡入淡出效果
2013/05/11 Javascript
实例分析javascript中的call()和apply()方法
2014/11/28 Javascript
jQuery中prevAll()方法用法实例
2015/01/08 Javascript
JS实现同时搜索百度和必应的方法
2015/01/27 Javascript
解决jQuery uploadify在非IE核心浏览器下无法上传
2015/08/05 Javascript
基于JavaScript实现移动端TAB触屏切换效果
2015/10/20 Javascript
跟我学习JScript的Bug与内存管理
2015/11/18 Javascript
Ajax和Comet技术总结
2017/02/19 Javascript
vue-prop父组件向子组件进行传值的方法
2018/03/01 Javascript
Vue路由history模式解决404问题的几种方法
2018/09/29 Javascript
微信小程序表单验证WxValidate的使用
2019/11/27 Javascript
jQuery实现数字华容道小游戏(实例代码)
2020/01/16 jQuery
在RedHat系Linux上部署Python的Celery框架的教程
2015/04/07 Python
详解Django缓存处理中Vary头部的使用
2015/07/24 Python
python检测服务器端口代码实例
2019/08/31 Python
Django模板标签{% for %}循环,获取制定条数据实例
2020/05/14 Python
在Keras中实现保存和加载权重及模型结构
2020/06/15 Python
Python常用外部指令执行代码实例
2020/11/05 Python
Python利用imshow制作自定义渐变填充柱状图(colorbar)
2020/12/10 Python
css3翻牌翻数字的示例代码
2020/02/07 HTML / CSS
HTML5新表单元素_动力节点Java学院整理
2017/07/12 HTML / CSS
白俄罗斯女装和针织品网上商店:Presli.by
2019/10/13 全球购物
戴尔新西兰官网:Dell New Zealand
2020/01/07 全球购物
Muziker英国:中欧最大的音乐家商店
2020/02/05 全球购物
数据库笔试题
2013/05/09 面试题
大学自主招生推荐信
2014/05/10 职场文书
董事长秘书岗位职责
2015/02/13 职场文书
2015年“7.11”世界人口日宣传活动方案
2015/05/06 职场文书
机关干部作风整顿心得体会
2016/01/22 职场文书
2016年乡镇综治宣传月活动总结
2016/03/16 职场文书