再谈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 顺便学习下CSS选择器 奇偶匹配nth-child(even)
May 24 Javascript
JavaScript的各种常见函数定义方法
Sep 16 Javascript
jQuery选择器querySelector的使用指南
Jan 23 Javascript
超精准的javascript验证身份证号的具体实现方法
Nov 18 Javascript
三种Node.js写文件的方式
Mar 08 Javascript
使用Vue.js创建一个时间跟踪的单页应用
Nov 28 Javascript
JavaScript实现form表单的多文件上传
Mar 27 Javascript
基于JavaScript实现新增内容滚动播放效果附完整代码
Aug 24 Javascript
bootstrap模态框关闭后清除模态框的数据方法
Aug 10 Javascript
微信小程序云开发详细教程
May 16 Javascript
JavaScript闭包相关知识解析
Oct 19 Javascript
微信小程序在text文本实现多种字体样式
Nov 08 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多种形式发送邮件(mail qmail邮件系统 phpmailer类)
2014/01/22 PHP
php汉字转拼音的示例
2014/02/27 PHP
php.ini中的request_order推荐设置
2015/05/10 PHP
php中文字符串截取多种方法汇总
2016/10/06 PHP
php获取远程图片并下载保存到本地的方法分析
2016/10/08 PHP
PHP实现的多维数组去重操作示例
2018/07/21 PHP
Javascript MD4
2006/12/20 Javascript
jQuery 源码分析笔记(3) Deferred机制
2011/06/19 Javascript
javascript 随机展示头像实现代码
2011/12/06 Javascript
jQuery实现鼠标移到元素上动态提示消息框效果
2013/10/20 Javascript
Javascript浅谈之引用类型
2013/12/18 Javascript
引用外部脚本时script标签关闭的写法
2014/01/20 Javascript
Javascript 中创建自定义对象的方法汇总
2014/12/04 Javascript
深入理解JavaScript系列(41):设计模式之模板方法详解
2015/03/04 Javascript
js数组的五种迭代方法及两种归并方法(推荐)
2016/06/14 Javascript
JavaScript中的call和apply的用途以及区别
2017/01/11 Javascript
JS中Select下拉列表类(支持输入模糊查询)功能
2017/01/17 Javascript
Vue表单验证插件Vue Validator使用方法详解
2017/04/07 Javascript
JS跳转手机站url的若干注意事项
2017/10/18 Javascript
全新打包工具parcel零配置vue开发脚手架
2018/01/11 Javascript
微信小程序自定义组件实现tabs选项卡功能
2018/07/14 Javascript
Python的Bottle框架中获取制定cookie的教程
2015/04/24 Python
Python简单实现自动删除目录下空文件夹的方法
2017/08/29 Python
python先序遍历二叉树问题
2017/11/10 Python
使用python 和 lint 删除项目无用资源的方法
2017/12/20 Python
Python常用字符串替换函数strip、replace及sub用法示例
2018/05/21 Python
Python字符串的全排列算法实例详解
2019/01/07 Python
Python二次规划和线性规划使用实例
2019/12/09 Python
检测tensorflow是否使用gpu进行计算的方式
2020/02/03 Python
全球性的在线婚纱礼服工厂:27dress.com
2019/03/21 全球购物
教师自荐信
2013/12/10 职场文书
安全责任协议书
2014/04/21 职场文书
2014旅游局领导班子四风问题对照检查材料思想汇报
2014/09/19 职场文书
张丽莉事迹观后感
2015/06/16 职场文书
孟佩杰观后感
2015/06/17 职场文书
vue 数字翻牌器动态加载数据
2022/04/20 Vue.js