再谈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 相关文章推荐
使用jquery与图片美化checkbox和radio控件的代码(打包下载)
Nov 11 Javascript
jquery lazyload延迟加载技术的实现原理分析
Jan 24 Javascript
js页面跳转的问题(跳转到父页面、最外层页面、本页面)
Aug 14 Javascript
在jQuery中处理XML数据的大致方法
Aug 14 Javascript
JS实现商品筛选功能
Aug 19 Javascript
es6 字符串String的扩展(实例讲解)
Aug 03 Javascript
薪资那么高的Web前端必看书单
Oct 13 Javascript
Vue 过滤器filters及基本用法
Dec 26 Javascript
vue webpack打包后图片路径错误的完美解决方法
Dec 07 Javascript
3分钟了解vue数据劫持的原理实现
May 01 Javascript
nuxt框架中对vuex进行模块化设置的实现方法
Sep 06 Javascript
vue实现将一个数组内的相同数据进行合并
Nov 07 Javascript
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
PHP获取客户端真实IP地址的5种情况分析和实现代码
2014/07/08 PHP
PHP把MSSQL数据导入到MYSQL的方法
2014/12/27 PHP
Yii2框架制作RESTful风格的API快速入门教程
2016/11/08 PHP
php实现微信和支付宝支付的示例代码
2020/08/11 PHP
禁止JQuery中的load方法装载IE缓存中文件的方法
2009/09/11 Javascript
jQuery ajax BUG:object doesn't support this property or method
2010/07/06 Javascript
javascript中的原型链深入理解
2014/02/24 Javascript
初识SmartJS - AOP三剑客
2014/06/08 Javascript
js实现获取当前时间是本月第几周的方法
2015/08/11 Javascript
基于jquery实现页面滚动时顶部导航显示隐藏
2020/04/20 Javascript
AngularJS基础 ng-dblclick 指令用法
2016/08/01 Javascript
深入浅出讲解ES6的解构
2016/08/03 Javascript
微信小程序 MD5加密登录密码详解及实例代码
2017/01/12 Javascript
浅谈node.js 命令行工具(cli)
2018/05/10 Javascript
layer.open 按钮的点击事件关闭方法
2018/08/17 Javascript
js设置默认时间跨度过程详解
2019/07/17 Javascript
[02:54]辉夜杯主赛事第二日败者组 iG.V赛后采访
2015/12/26 DOTA
详解python并发获取snmp信息及性能测试
2017/03/27 Python
Python IDLE入门简介
2017/12/08 Python
python消除序列的重复值并保持顺序不变的实例
2018/11/08 Python
Pytorch在dataloader类中设置shuffle的随机数种子方式
2020/01/14 Python
DRF框架API版本管理实现方法解析
2020/08/21 Python
GAP欧盟网上商店:GAP EU
2016/09/13 全球购物
SAZAC的动物连体衣和动物睡衣:Kigurumi Shop
2020/03/14 全球购物
介绍一下except的用法和作用
2015/01/22 面试题
妇产科护士自我鉴定
2013/10/15 职场文书
应届生财务管理求职信
2013/11/06 职场文书
幼儿教师研修感言
2014/02/12 职场文书
高三上学期学习自我评价
2014/04/23 职场文书
租房协议书范文
2014/08/20 职场文书
2014年体育教学工作总结
2014/12/09 职场文书
导盲犬小Q观后感
2015/06/11 职场文书
2016年大学迎新工作总结
2015/10/14 职场文书
《折线统计图》教学反思
2016/02/22 职场文书
opencv 分类白天与夜景视频的方法
2021/06/05 Python
mysql中DCL常用的用户和权限控制
2022/03/31 MySQL