前言

对于前端开发者来说,多线程是一个比较陌生的话题。因为JavaScript是单线程语言。也就是说,所有任务只能在一个线程上完成,一次只能做一件事。前面的任务没做完,后面的任务只能等着。
UI渲染与JavaScript是共同使用主线程。如果JavaScript运行过长,可能就会中断UI渲染,从而导致页面卡顿。
为此,JavaScript推出了异步的处理方法。但终归到底还是单线程的。而且随着电脑计算能力的增强,尤其是多核CPU的出现,单线程带来很大的不便,无法充分发挥计算机的计算能力。
Web Workers就应运而生了。通过使用Web Workers,Web应用程序可以在独立于主线程的后台线程中,运行一个脚本操作。这样做的好处是可以在独立线程中执行费时的处理任务,主线程从而不会因此被阻塞。

多线程

我们从熟悉的领域入手来了解多线程是什么。
通常从事开发的同学都对计算机配置有一定的了解,知道CPU配置中都标明几核几线程。比如说6核12线程、8核16线程等。
(电脑比较渣,只有2核4线程)
一般来说:一个CPU有几核就可以跑几个线程,比如说二核二线程--说明这个CPU同时最多能够运行两个线程,而二核四线程是使用了超线程技术,使得单个核像有两个核一样,速度要比二核二线程要快。
了解了基本信息之后,来看看JavaScript的多线程--web workers。

web workers

HTML5引入了Web Workers,让JavaScript支持多线程。接下来用Web Workers做一个斐波那契函数来举例:
// worker.js
function fibonacci(n) {
function fib(n, v1, v2) {
if (n == 1)
return v1;
if (n == 2)
return v2;
else
return fib(n - 1, v2, v1 + v2);
}
return fib(n, 1, 1)
} // 通过onmessage回调函数接收主线程的数据
onmessage = function (e) {
// 通过e.data接收从主线程中传过来的数据。
var num = e.data;
var result = fibonacci(num);
// 通过postMessage向主线程传输结果。
postMessage(result);
}

把这个函数写到worker.js中。接着在index.js文件中使用new关键字,调用Worker()构造函数,新建一个Worker线程。
var worker = new Worker('worker.js文件的url');

worker.onmessage = function (e) {
console.log("result: " + e.data);
}
worker.postMessage(100); worker.terminate();

Worker()构造函数的参数是刚刚定义的worker脚本文件。但是Worker构造函数不能读取本地文件,所以这个脚本必须来自网络。然后,调用worker.postMessage()方法,向Worker发消息。最后,主通过worker.onmessage指定监听函数,接收Worker发回来的消息。Worker完成任务以后,就可以把它关掉。

数据通信

主线程与Worker之间的通信内容,可以是基本类型的,也可以是引用类型的,而且是通过值传递的。Worker对通信内容的修改,不会影响到主线程。事实上,浏览器内部的运行机制是,先将通信内容通过类似JSON.stringify()的api将内容转为字符串,再传给Worker,后者将其还原。
正如所想的那样,这种方式会造成性能问题。当主线程向Worker发送几百上千兆大小的文件,默认情况下浏览器会将其拷贝一份。为了解决这个问题,JavaScript允许主线程通过TransferableObjects方法把数据直接转移给Worker线程,但是一旦传输了,主线程就再也无法使用这些数据了。这是为了防止出现多个线程同时处理数据的风险。
// Transferable Objects 格式
worker.postMessage(arrayBuffer, [arrayBuffer]); // 例子
var ab = new ArrayBuffer(1);
worker.postMessage(ab, [ab]);

内联Web Worker

一般来说,Web Worker的载入是一个单独的JavaScript脚本文件,但也是可以与主线程用一个文件中载入:
<!DOCTYPE html>
<body>
<script id='worker' type='app/worker'>
function fibonacci(n) {
...
} // 通过onmessage回调函数接收主线程的数据
onmessage = function (e) {
// 通过e.data接收从主线程中传过来的数据。
var num = e.data;
var result = fibonacci(num);
// 通过postMessage向主线程传输结果。
postMessage(result);
}
</script>
</body>
</html>

必须指定<script>标签的type属性是一个浏览器不认识的值,比如app/worker
然后,读取这段嵌入页面的脚本,用Worker来处理
var blob = new Blob([document.querySelector('#worker').textContent]);
var url = window.URL.createObjectURL(blob);
var worker = new Worker(url); worker.onmessage = function (e) {
console.log("result: " + e.data);
}
worker.postMessage(100);

先将嵌入网页的脚本代码,转为一个二进制对象,然后为这个二进制对象生成URL,再让Worker加URL。done

线程同步

最后来说说JavaScript版本的线程同步。
由于Web Workers是不可以操作DOM的,因为同一个DOM节点只能有一个线程操作,不允许同一个变量或者内存被同时写入。
如果web Workers可以操作DOM呢?那会怎么样,当然是要用到线程同步的方式限制线程写入。
同步的意思是协同、互相配合的意思,按照预定的先后次序进行运行。比如说线程A和线程B同步,A执行到一定程度时要依赖B的某个运行结果,那么就必须先停下来,让B运行,B运行完后,把结果给到A,A再继续操作。
线程同步主要是靠锁来实现的,可以分为以下3种:
「互斥锁」
var mutext = new Mutext();
function changeDOM (style) {
mutext.lock();
document.getElementById('app').style = style;
mutext.unlock();
} // worker1
changeStyle({width: 100px}); // worker2
changeStyle({width: 150px});

在改变某个DOM元素的样式时,先把这部分代码的执行给锁住了,只有执行完了才释放这把锁,其他线程运行到这时也要去申请那把锁,但是由于这把锁没有被释放,所以它就阻塞在那里,只有等到锁被释放了,它才能拿到这把锁再继续加锁。
互斥锁使用太多会导致性能下降,因为线程阻塞在那里而且还要不断的检测锁能不能用,所以要占用CPU。
「读写锁」
var rwLock = new ReadWriteLock();
function changeStyle (style) {
rwLock.writeLock();
document.getElementById('app').style = style;
rwLock.unlock();
} function getStyle () {
rwLock.readLock();
var style = document.getElementById('app').style
rwLock.unlock();
return style;
}

在第二个函数getStyle()获取样式时可以给它加一个读锁,这样其他线程如果想读是可以同时读的,但是不允许有一个线程写入。如果有线程调用了第一个函数,那么调用第二个函数的线程都会被阻塞,因为在写的过程中,不运行被读取。
「条件变量」
条件变量是为解决生产者和消费者的问题,由于互斥锁和读写锁会导致线程一直阻塞而且占用CPU,而使用信号通知的方式可以先让阻塞的线程进入睡眠状态,等生产者生产出东西后通知消费者,再唤醒它进行消费。
然而现实上JavaScript是没有线程同步的概念。因为webWorker是无法操作DOM,也没有window对象,每个线程的数据都是独立的。前面说过是通过拷贝复制的方式传递的。所以不存在共享同一块内存区域。
 
作者: zhangwinwin
来源:github

JavaScript与多线程的不解之缘!的更多相关文章

  1. JavaScript可否多线程? 深入理解JavaScript定时机制(转载)

    说明:最近写 js 时需要用setinterval函数做定时操作,谁知道,刚开始后运行完好,但一段时间后他就抽风了,定时任务运行的时间间隔越来越短,频率加快,这是一个完全不能容忍的问题,带着一个可以出 ...

  2. 【JavaScript】吃饱了撑的系列之JavaScript模拟多线程并发

    前言 最近,明学是一个火热的话题,而我,却也想当那么一回明学家,那就是,把JavaScript和多线程并发这两个八竿子打不找的东西,给硬凑了起来,还写了一个并发库concurrent-thread-j ...

  3. JavaScript可否多线程? 深入理解JavaScript定时机制

    JavaScript的setTimeout与setInterval是两个很容易欺骗别人感情的方法,因为我们开始常常以为调用了就会按既定的方式执行, 我想不少人都深有同感, 例如 setTimeout( ...

  4. JavaScript 编写多线程代码引用Concurrent.Thread.js(转)

    这是一个很简单的功能实现: <script type="text/javascript" src="Concurrent.Thread.js">&l ...

  5. javascript实现多线程 Concurrent.Thread.js

    在这次我的项目中,因为前端要检测硬件加载并识别,再向后台请求发送数据,然后再返回的相应的配置文件!在这过程,要好时好几秒钟,严重影响体验效果,所以在网上靠看的方案,运用多线程去处理,这效果明显改善! ...

  6. js javascript 实现多线程

    在讲之前,大家都知道js是基于单线程的,而这个线程就是浏览器的js引擎. 首先来看一下大家用的浏览器都具有那些线程吧. 假如我们要执行一些耗时的操作,比如加载一张很大的图片,我们可能需要一个进度条来让 ...

  7. javascript实现多线程提升项目加载速度

    以前大家都认为js是单线程执行的,假如我们要执行一些耗时的操作,比如加载一张很大的图片,我们可能需要一个进度条来让用户进行等待,在等待的过程中,整个js线程会被阻塞,后面的代码不能正常运行,这可能大大 ...

  8. Javascript多线程引擎(二)

    多线程Javascript解释器的大致架构 由于一个完整的解释器类似Google V8的解释器需要的工作量非常的大如需要实现如下的模块: 词法分析,语法分析器,AST转Byte模块,解释执行模块和JI ...

  9. 转载:JavaScript多线程编程简介

    虽然有越来越多的网站在应用AJAX技术进行开发,但是构建一个复杂的AJAX应用仍然是一个难题.造成这些困难的主要原因是什么呢?是与服务器的异步通信问题?还是GUI程序设计问题呢?通常这两项工作都是由桌 ...

随机推荐

  1. 将从数据库查询出来的带有父子结构的list转换成treeList结构

    package test; import java.sql.Connection; import java.sql.DriverManager; import java.sql.PreparedSta ...

  2. JDBC删除

    1 if(conn != null){ 2 String temps="3"; 3 conn.setAutoCommit(false); 4 PreparedStatement p ...

  3. Linux 时间同步 03 ntpdate时间同步

    Linux 时间同步 03 ntpdate时间同步 目录 Linux 时间同步 03 ntpdate时间同步 安装ntpdate 修改/etc/sysconfig/ntpdate 使用ntpdate手 ...

  4. Asp.net Core 2.0 实现Cookie会话

    与1.0版本相比微软做了一些调整.详细请参考官方文档,我这里就讲2.0的吧 1.首先要在 根目录下 Startup.cs 类中启用 cookie会话,有两处要配置 第一处在  public void ...

  5. 11. const 修饰成员函数

    const 限定只读,对函数的实参进行保护 常数据成员:必须出现在类的定义体中,常数据成员必须进行初始化,并且不能被更新,但常数据成员的初始化只能通过构造函数的初始化列表进行 1. 常函数 成员函数加 ...

  6. 微信小程序--仿微信小程序朋友圈Pro(内容发布、点赞、评论、回复评论)

    微信小程序--仿微信小程序朋友圈Pro(内容发布.点赞.评论.回复评论) 项目开源地址M朋友圈Pro 求个Star 项目背景 ​ 基于原来的开源项目 微信小程序仿朋友圈功能开发(发布.点赞.评论等功能 ...

  7. mysql使用全文索引实现大字段的模糊查询

    0.场景说明 centos7 mysql5.7 InnoDB引擎 0.1创建表 DROP TABLE IF EXISTS tbl_article_content; CREATE TABLE tbl_a ...

  8. 地图开发笔记(一):百度地图介绍、使用和Qt内嵌地图Demo

    前言   Qt在地图方面的研发.   百度地图 介绍   百度的地图分为多个开发,都是在线的(离线的需要自己提取,本篇解说在线地图).  百度地图JavaScript API支持HTTP和HTTPS, ...

  9. Linux下的strip命令学习

    strip strip是Linux下的一个命令.可以用于给应用脱衣服,帮助我们抹除一些调试信息.(虽然不知道具体是什么,但是会用就好了) 在嵌入式开发领域用到的应该比较多 首先,先写一个示例看看 // ...

  10. [从源码学设计]蚂蚁金服SOFARegistry之续约和驱逐

    [从源码学设计]蚂蚁金服SOFARegistry之续约和驱逐 目录 [从源码学设计]蚂蚁金服SOFARegistry之续约和驱逐 0x00 摘要 0x01 业务范畴 1.1 失效剔除 1.2 服务续约 ...