百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术分析 > 正文

Worker在JavaScript中发挥了什么作用?

liebian365 2024-10-21 08:42 27 浏览 0 评论

全文共2548字,预计学习时长8分钟



多年以来,JavaScript最麻烦的特征之一就是,如果某个任务耗时太长,剩下的代码就会遇到阻塞而无法运行。


JavaScript是单线程的编程语言,这一特点导致用户需要等待代码按照顺序运行。


但事实上有一种方式可以避免这种困境,它就是Worker。


在本文中,小芯将介绍Worker的使用方式。


阅读之前


读者需要知道JavaScript是一种单线程语言。


同步编码


先来看看读者可能已经熟知的JavaScript代码样式


let cnt = 0;
for (let i = 0; i < 10e8; i += 1) {
  cnt += 1;
}
console.log(cnt);


在这段代码的for循环中,cnt每次增加1,总共增加了次。而console.log要等到for循环结束后才能执行。


在Chrome浏览器控制台中,这需要花费很长时间。



这段代码需要耗费将近3秒。


无论如何,直到for循环结束cnt才会打印出来。


许多开发者都被这个问题困扰,因为用户必须等待当前执行的任务完成。


异步代码



JavaScript的设计者为开发者提供了不会阻塞程序流的特殊函数。这些函数处于和普通任务不同的等待队列中。一般而言,它们可以在所有普通程序执行之后再执行。这些任务被称为异步任务。


let cnt = 0;
setTimeout(() => {
  for (let i = 0; i < 10e8; i += 1) {
    cnt += 1;
  }
  console.log(cnt);
});
console.log(cnt);


这段代码的运行结果和前面一段有所不同。



但仍然需要等待较长的时间。


这里的for循环包含在 setTimeout中. setTimeout中的代码会等待所有普通任务完成后再执行。


但这并非解决代码流堵塞问题的最佳方案。虽然 setTimeout是一个不阻碍正常代码流的异步函数,但是这样做仅仅改变了函数的执行顺序。


let cnt = 0;
setTimeout(() => {
  for (let i = 0; i < 10e8; i += 1) {
    cnt += 1;
  }
  console.log(cnt);
});
setTimeout(() => {
  console.log(cnt);
});
console.log(cnt);


看看这个例子,其中使用了另一个setTimeout,它包含一个立即打印cnt的函数。然而第二个setTimeout总是在第一个setTimeout结束for循环后才开始运行console.log(cnt)指令。如果第一个setTimeout包含的任务耗时较长,那么第二个setTimeout将无法运行。


为什么?因为Javascript是一种单线程编程语言。异步函数存在于不同的任务队列中,但它们仍然遵循单线程规则。


Workers


Web Worker是一种网络接口,这意味着它无法访问或管理文档对象模型。Worker存在于一个不同的线程中,它和主线程互不干扰。它在一个新的Worker对象创建时接受信息,然后向worker发送信息。


let worker;
if ('Worker' in window) {
  worker = new Worker('file_name');
}


创建新的Worker实例十分简单。new Worker这条指令接受一个字符串或者超链接作为参数。这个参数的格式一般如下所示:


new Worker('/worker.js');


实例创建之后,可以向另一个线程发送信息。


worker.postMessage('From Main Thread');


于是可以在worker所处的线程中收到这个信息。


// worker.jsthis.addEventListener('message', event => {
  console.log(event.data);
});



Worker成功接受信息。


如果仔细查看日志,会发现文件名是worker.js而不是main.js或者app.js。这说明worker接受了完好的信息。


现在,让worker在收到主线程信息的同时也向主线程发送一条信息。


// main.js
worker.addEventListener('message', event => {
  console.log(event.data);
});// worker.js
this.addEventListener('message', event => {
  ...
  this.postMessage('From Worker Thread');
});


这段代码看起来有些冗余。读者将会看到其工作流,但是首先来看看结果。



现在能看到两条信息。


这段代码是如何工作的呢?



onMessage代表从线程另一端接受信息这一事件,postMessage则代表向线程另一端发送信息这一事件。


代码测试



在这个测试中可以看到两点。


· For循环的运行总时长

· 代码阻断


这个示例使用React制作。状态消息应该按照预期打印,然而同步和异步行为不会打印除了总运行时长外的任何信息。因为在React中,状态改变也是一种异步行为,需要等待普通任务和其他前序异步任务完成才能执行。


另一方面,worker不需要等待,因为它位于另一个线程中。For循环在worker线程中,异步任务则在主线程中,所以它们不会互相打断。


结论



通常worker被用于占用大量CPU资源的程序中,比如2D canvas 和矢量图。因为worker位于另一个线程中,它不会阻断主线程中的任何任务,比如UI渲染。如果能将worker运用自如,它的效果将十分强大。包括IE10在内的众多浏览器都能够很好地支持这一功能。


留言点赞关注

我们一起分享AI学习与发展的干货

如转载,请后台留言,遵守转载规范

相关推荐

“版本末期”了?下周平衡补丁!国服最强5套牌!上分首选

明天,酒馆战棋就将迎来大更新,也聊了很多天战棋相关的内容了,趁此机会,给兄弟们穿插一篇构筑模式的卡组推荐!老规矩,我们先来看10职业胜率。目前10职业胜率排名与一周前基本类似,没有太多的变化。平衡补丁...

VS2017 C++ 程序报错“error C2065:“M_PI”: 未声明的标识符&quot;

首先,程序中头文件的选择,要选择头文件,在文件中是没有对M_PI的定义的。选择:项目——>”XXX属性"——>配置属性——>C/C++——>预处理器——>预处理器定义,...

东营交警实名曝光一批酒驾人员名单 88人受处罚

齐鲁网·闪电新闻5月24日讯酒后驾驶是对自己和他人生命安全极不负责的行为,为守护大家的平安出行路,东营交警一直将酒驾作为重点打击对象。5月23日,东营交警公布最新一批饮酒、醉酒名单。对以下驾驶人醉酒...

Qt界面——搭配QCustomPlot(qt platform)

这是我第一个使用QCustomPlot控件的上位机,通过串口精确的5ms发送一次数据,再将读取的数据绘制到图表中。界面方面,尝试卡片式设计,外加QSS简单的配了个色。QCustomPlot官网:Qt...

大话西游2分享赢取种族坐骑手办!PK趣闻录由你书写

老友相聚,仗剑江湖!《大话西游2》2021全民PK季4月激燃打响,各PK玩法鏖战齐开,零门槛参与热情高涨。PK季期间,不仅各种玩法奖励丰厚,参与PK趣闻录活动,投稿自己在PK季遇到的趣事,还有机会带走...

测试谷歌VS Code AI 编程插件 Gemini Code Assist

用ClaudeSonnet3.7的天气测试编码,让谷歌VSCodeAI编程插件GeminiCodeAssist自动编程。生成的文件在浏览器中的效果如下:(附源代码)VSCode...

顾爷想知道第4.5期 国服便利性到底需优化啥?

前段时间DNF国服推出了名为“阿拉德B计划”的系列改版计划,截至目前我们已经看到了两项实装。不过关于便利性上,国服似乎还有很多路要走。自从顾爷回归DNF以来,几乎每天都在跟我抱怨关于DNF里面各种各样...

掌握Visual Studio项目配置【基础篇】

1.前言VisualStudio是Windows上最常用的C++集成开发环境之一,简称VS。VS功能十分强大,对应的,其配置系统较为复杂。不管是对于初学者还是有一定开发经验的开发者来说,捋清楚VS...

还嫌LED驱动设计套路深?那就来看看这篇文章吧

随着LED在各个领域的不同应用需求,LED驱动电路也在不断进步和发展。本文从LED的特性入手,推导出适合LED的电源驱动类型,再进一步介绍各类LED驱动设计。设计必读:LED四个关键特性特性一:非线...

Visual Studio Community 2022(VS2022)安装图文方法

直接上步骤:1,首先可以下载安装一个VisualStudio安装器,叫做VisualStudioinstaller。这个安装文件很小,很快就安装完成了。2,打开VisualStudioins...

Qt添加MSVC构建套件的方法(qt添加c++11)

前言有些时候,在Windows下因为某些需求需要使用MSVC编译器对程序进行编译,假设我们安装Qt的时候又只是安装了MingW构建套件,那么此时我们该如何给现有的Qt添加一个MSVC构建套件呢?本文以...

Qt为什么站稳c++GUI的top1(qt c)

为什么现在QT越来越成为c++界面编程的第一选择,从事QT编程多年,在这之前做C++界面都是基于MFC。当时为什么会从MFC转到QT?主要原因是MFC开发界面想做得好看一些十分困难,引用第三方基于MF...

qt开发IDE应该选择VS还是qt creator

如果一个公司选择了qt来开发自己的产品,在面临IDE的选择时会出现vs或者qtcreator,选择qt的IDE需要结合产品需求、部署平台、项目定位、程序猿本身和公司战略,因为大的软件产品需要明确IDE...

Qt 5.14.2超详细安装教程,不会来打我

Qt简介Qt(官方发音[kju:t],音同cute)是一个跨平台的C++开库,主要用来开发图形用户界面(GraphicalUserInterface,GUI)程序。Qt是纯C++开...

Cygwin配置与使用(四)——VI字体和颜色的配置

简介:VI的操作模式,基本上VI可以分为三种状态,分别是命令模式(commandmode)、插入模式(Insertmode)和底行模式(lastlinemode),各模式的功能区分如下:1)...

取消回复欢迎 发表评论: