再谈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代码混淆综合解决方案-Javascript在线混淆器
Dec 18 Javascript
jquery 模拟雅虎首页的点击对话框效果
Apr 11 Javascript
初窥JQuery(二)事件机制(2)
Dec 06 Javascript
jquery isType() 类型判断代码
Feb 14 Javascript
javaScript让文本框内的最后一个文字的后面获得焦点实现代码
Jan 06 Javascript
jquery Easyui快速开发总结
Aug 20 Javascript
详解angular ui-grid之过滤器设置
Jun 07 Javascript
如何用RxJS实现Redux Form
Dec 29 Javascript
使用Jenkins部署React项目的方法步骤
Mar 11 Javascript
深入学习js函数的隐式参数 arguments 和 this
Jun 24 Javascript
vuex actions异步修改状态的实例详解
Nov 06 Javascript
Vue+Flask实现图片传输功能
Apr 01 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
Mysql中limit的用法方法详解与注意事项
2008/04/19 PHP
ThinkPHP+EasyUI之ComboTree中的会计科目树形菜单实现方法
2017/06/09 PHP
Phpstorm+Xdebug断点调试PHP的方法
2018/05/14 PHP
javascript 导出数据到Excel(处理table中的元素)
2009/12/18 Javascript
firefox下jquery iframe刷新页面提示会导致重复之前动作
2012/12/17 Javascript
javascript四舍五入函数代码分享(保留后几位)
2013/12/10 Javascript
CSS中position属性之fixed实现div居中
2015/12/14 Javascript
js鼠标按键事件和键盘按键事件用法实例汇总
2016/10/03 Javascript
jQuery动态创建元素以及追加节点的实现方法
2016/10/20 Javascript
JavaScript模块模式实例详解
2017/10/25 Javascript
在vue中实现简单页面逆传值的方法
2017/11/27 Javascript
微信小程序事件 bindtap bindinput代码实例
2019/08/26 Javascript
python创建和删除目录的方法
2015/04/29 Python
实例讲解Python的函数闭包使用中应注意的问题
2016/06/20 Python
Python爬虫设置代理IP的方法(爬虫技巧)
2018/03/04 Python
Python Flask前后端Ajax交互的方法示例
2018/07/31 Python
python 列表递归求和、计数、求最大元素的实例
2018/11/28 Python
pandas的to_datetime时间转换使用及学习心得
2019/08/11 Python
浅析python表达式4+0.5值的数据类型
2020/02/26 Python
Win 10下Anaconda虚拟环境的教程
2020/05/18 Python
基于python检查矩阵计算结果
2020/05/21 Python
python里glob模块知识点总结
2021/01/05 Python
微信小程序实现可实时改变转速的css3旋转动画实例代码
2018/09/11 HTML / CSS
CSS实现圆形放大镜狙击镜效果 只有圆圈里的放大
2012/12/10 HTML / CSS
全球知名的婚恋交友网站:Match.com
2017/01/05 全球购物
兰蔻美国官网:Lancome美国
2017/04/25 全球购物
美国最大的香水出口:FragranceX.com
2017/11/04 全球购物
程序员跳槽必看面试题总结
2013/06/28 面试题
历史教育专业个人求职信
2013/12/13 职场文书
党校培训自我鉴定范文
2014/04/10 职场文书
珍爱生命演讲稿
2014/05/10 职场文书
学校副校长四风对照检查材料整改措施
2014/09/25 职场文书
重阳节慰问信
2015/02/15 职场文书
参加招聘会后的感想
2015/08/10 职场文书
致创业的您:这类人不适合餐饮创业
2019/08/19 职场文书
python区块链实现简版工作量证明
2022/05/25 Python