再谈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 相关文章推荐
静态页面的值传递(三部曲)
Sep 25 Javascript
javascript实现日历控件(年月日关闭按钮)
Dec 12 Javascript
JS+CSS实现的蓝色table选项卡效果
Oct 08 Javascript
javascript对象的创建和访问
Mar 08 Javascript
JavaScript使用ZeroClipboard操作剪切板
May 10 Javascript
Node 自动化部署的方法
Oct 17 Javascript
Vue2.0学习系列之项目上线的方法步骤(图文)
Sep 25 Javascript
Node.js系列之安装配置与基本使用(1)
Aug 30 Javascript
微信小程序实现轨迹回放的示例代码
Dec 13 Javascript
ES6中的类(Class)示例详解
Dec 09 Javascript
vue实现拖拽进度条
Mar 01 Vue.js
vue实现省市区联动 element-china-area-data插件
Apr 22 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
php输出表格的实现代码(修正版)
2010/12/29 PHP
PHP防CC攻击实现代码
2011/12/29 PHP
PHP获取当前完整URL地址的函数
2014/12/21 PHP
Yii框架实现的验证码、登录及退出功能示例
2017/05/20 PHP
PHP实现数据库统计时间戳按天分组输出数据的方法
2017/10/10 PHP
PHP 使用二进制保存用户状态的实例
2018/01/29 PHP
PHP网站常见安全漏洞,及相应防范措施总结
2021/03/01 PHP
JavaScript 模拟用户单击事件
2009/12/31 Javascript
JQuery扩展插件Validate 5添加自定义验证方法
2011/09/05 Javascript
Jquery加载时从后台读取数据绑定到dropdownList实例
2013/06/09 Javascript
JavaScript动态修改背景颜色的方法
2015/04/16 Javascript
jQuery Validate初步体验(一)
2015/12/12 Javascript
JavaScript对象数组排序实例方法浅析
2016/06/15 Javascript
BootStrap使用popover插件实现鼠标经过显示并保持显示框
2016/06/23 Javascript
AngularJS入门教程之数据绑定原理详解
2016/11/02 Javascript
jquery实现拖动效果(代码分享)
2017/01/25 Javascript
微信小程序商城项目之淘宝分类入口(2)
2017/04/17 Javascript
nodejs搭建本地服务器轻松解决跨域问题
2018/03/21 NodeJs
layui的表单提交以及验证和修改弹框的实例
2019/09/09 Javascript
uni-app 自定义底部导航栏的实现
2020/12/11 Javascript
jQuery实现增删改查
2020/12/22 jQuery
[04:40]2016国际邀请赛中国区预选赛全程TOP10镜头集锦
2016/07/01 DOTA
python获取文件版本信息、公司名和产品名的方法
2014/10/05 Python
Python中使用wxPython开发的一个简易笔记本程序实例
2015/02/08 Python
从局部变量和全局变量开始全面解析Python中变量的作用域
2016/06/16 Python
Python实现简单网页图片抓取完整代码实例
2017/12/15 Python
Python Pandas数据中对时间的操作
2019/07/30 Python
python如何将多个PDF进行合并
2019/08/13 Python
基于Python爬取爱奇艺资源过程解析
2020/03/02 Python
Python pandas 列转行操作详解(类似hive中explode方法)
2020/05/18 Python
如何基于Python和Flask编写Prometheus监控
2020/11/25 Python
保险经纪人求职信
2014/03/11 职场文书
团队激励口号
2014/06/06 职场文书
pandas数值排序的实现实例
2021/07/25 Python
一文搞懂Python Sklearn库使用
2021/08/23 Python
能用CSS实现的就不要麻烦JavaScript了
2021/10/05 HTML / CSS