再谈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 相关文章推荐
为原生js Array增加each方法
Apr 07 Javascript
node.js中的fs.lchownSync方法使用说明
Dec 16 Javascript
所见即所得的富文本编辑器bootstrap-wysiwyg使用方法详解
May 27 Javascript
RequireJS 依赖关系的实例(推荐)
Jan 21 Javascript
JS中实现函数return多个返回值的实例
Feb 21 Javascript
vue-hook-form使用详解
Apr 07 Javascript
layui表格实现代码
May 20 Javascript
vue实现仿淘宝结账页面实例代码
Nov 08 Javascript
详解Ant Design of React的安装和使用方法
Dec 27 Javascript
如何根据业务封装自己的功能组件
Apr 19 Javascript
Webpack按需加载打包chunk命名的方法
Sep 22 Javascript
处理canvas绘制图片模糊问题
May 11 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  实现等比压缩图片尺寸和大小实例代码
2016/10/08 PHP
ThinkPHP框架实现的微信支付接口开发完整示例
2019/04/10 PHP
在 Laravel 6 中缓存数据库查询结果的方法
2019/12/11 PHP
静态的动态续篇之来点XML
2006/08/15 Javascript
ExtJS扩展 垂直tabLayout实现代码
2009/06/21 Javascript
js变换显示图片的实例
2013/04/16 Javascript
Jquery倒数计时按钮setTimeout的实例代码
2013/07/04 Javascript
node.js集成百度UE编辑器
2015/02/05 Javascript
javascript实现汉字转拼音代码分享
2015/04/20 Javascript
JS实现超简洁网页title标题跑动闪烁提示效果代码
2015/10/23 Javascript
BootStrap的alert提示框的关闭后再显示怎么解决
2016/05/17 Javascript
用js实现放大镜的效果的简单实例
2016/05/23 Javascript
JS实现动态增加和删除li标签行的实例代码
2016/10/16 Javascript
Vue.js中extend选项和delimiters选项的比较
2017/07/17 Javascript
基于Vue框架vux组件库实现上拉刷新功能
2017/11/28 Javascript
vue.js使用v-model指令实现的数据双向绑定功能示例
2018/05/22 Javascript
vue之封装多个组件调用同一接口的案例
2020/08/11 Javascript
[04:29]【TI9采访】OG.N0tail在胜者组决赛后接受采访
2019/08/25 DOTA
Python中处理时间的几种方法小结
2015/04/09 Python
Python实现处理管道的方法
2015/06/04 Python
查找python项目依赖并生成requirements.txt的方法
2018/07/10 Python
Python用类实现扑克牌发牌的示例代码
2020/06/01 Python
Keras自定义实现带masking的meanpooling层方式
2020/06/16 Python
Python unittest如何生成HTMLTestRunner模块
2020/09/08 Python
浅谈Python __init__.py的作用
2020/10/28 Python
收藏!10个免费高清视频素材网站!【设计、视频剪辑必备】
2021/03/18 杂记
css3 线性渐变和径向渐变示例附图
2014/04/08 HTML / CSS
使用phonegap进行提示操作的具体方法
2017/03/30 HTML / CSS
个人求职简历的自我评价
2013/10/19 职场文书
会计专业自荐信
2013/12/02 职场文书
试用期自我鉴定范文
2014/03/20 职场文书
交通安全寄语大全
2014/04/08 职场文书
2016年春节慰问信息
2015/03/25 职场文书
2015年社区民政工作总结
2015/04/21 职场文书
听课评课活动心得体会
2016/01/15 职场文书
ant design vue的form表单取值方法
2022/06/01 Vue.js