JavaScript与多线程的不解之缘!

2024-03-09 04:59

本文主要是介绍JavaScript与多线程的不解之缘!,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言

对于前端开发者来说,多线程是一个比较陌生的话题。因为JavaScript是单线程语言。也就是说,所有任务只能在一个线程上完成,一次只能做一件事。前面的任务没做完,后面的任务只能等着。

UI渲染与JavaScript是共同使用主线程。如果JavaScript运行过长,可能就会中断UI渲染,从而导致页面卡顿。

为此,JavaScript推出了异步的处理方法。但终归到底还是单线程的。而且随着电脑计算能力的增强,尤其是多核CPU的出现,单线程带来很大的不便,无法充分发挥计算机的计算能力。

Web Workers就应运而生了。通过使用Web Workers,Web应用程序可以在独立于主线程的后台线程中,运行一个脚本操作。这样做的好处是可以在独立线程中执行费时的处理任务,主线程从而不会因此被阻塞。

多线程

我们从熟悉的领域入手来了解多线程是什么。

通常从事开发的同学都对计算机配置有一定的了解,知道CPU配置中都标明几核几线程。比如说6核12线程、8核16线程等。

image

(电脑比较渣,只有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;elsereturn 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>

必须指定

然后,读取这段嵌入页面的脚本,用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').stylerwLock.unlock();return style;
}

在第二个函数getStyle()获取样式时可以给它加一个读锁,这样其他线程如果想读是可以同时读的,但是不允许有一个线程写入。如果有线程调用了第一个函数,那么调用第二个函数的线程都会被阻塞,因为在写的过程中,不运行被读取。

「条件变量」
条件变量是为解决生产者和消费者的问题,由于互斥锁和读写锁会导致线程一直阻塞而且占用CPU,而使用信号通知的方式可以先让阻塞的线程进入睡眠状态,等生产者生产出东西后通知消费者,再唤醒它进行消费。

然而现实上JavaScript是没有线程同步的概念。因为webWorker是无法操作DOM,也没有window对象,每个线程的数据都是独立的。前面说过是通过拷贝复制的方式传递的。所以不存在共享同一块内存区域。

作者: zhangwinwin
链接:JavaScript与多线程的不解之缘!
来源:github

这篇关于JavaScript与多线程的不解之缘!的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/789583

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

在cscode中通过maven创建java项目

在cscode中创建java项目 可以通过博客完成maven的导入 建立maven项目 使用快捷键 Ctrl + Shift + P 建立一个 Maven 项目 1 Ctrl + Shift + P 打开输入框2 输入 "> java create"3 选择 maven4 选择 No Archetype5 输入 域名6 输入项目名称7 建立一个文件目录存放项目,文件名一般为项目名8 确定