博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
JavaScript 单线程不简单.md
阅读量:6256 次
发布时间:2019-06-22

本文共 2796 字,大约阅读时间需要 9 分钟。

我们常听说 JavaScript 是单线程的,那这个单线程是什么意思呢?单线程是否意味 JavaScript 存在性能缺陷呢?

在浏览器端,JavaScript 单线程指的是 JavaScript 的执行线程与 UI 渲染线程共用一个线程。对于 NodeJS 而言,单线程指的是它的 JavaScript 执行线程是单线程。虽然 JavaScript 只能单线程执行,但 JavaScript 引擎可不是,它能够创建多个线程为主线程服务。Web Worker 已经得到大部分浏览器的支持,NodeJS 也拥有自己的线程池来处理 I/O 操作。无论前端还是后端,JavaScript 已经能够利用多个线程来提升程序性能了。

NodeJS 单线程异步 I/O 模型

NodeJS 是单线程异步 I/O 模型。换句话说,NodeJS 代码执行占用一个线程,而代码中的 I/O 操作则是交给其它线程执行,执行完毕后将结果交还给主线程。

对于 NodeJS 来说,单线程不仅不是劣势,它对于降低编程复杂度还有很重要的作用,单线程避免了多线程编程模型多线程死锁、状态同步等问题。而异步 I/O 避免了单线程同步编程模型的阻塞问题,使 CPU 得到更充分的使用。

NodeJS 异步 I/O 模型的实现离不开 libuv 层,libuv 提供了一个线程池来执行 I/O 操作,执行完毕后再将结果返回给执行线程,因此 I/O 操作不会阻塞执行线程地继续执行。libuv 是一个事件驱动的异步 I/O 库,它是跨平台的,在 *nix 平台下,自行实现了线程池,在 windows 平台采用了 IOCP,IOCP 内部仍是线程池原理,libuv 的线程池默认为 4 个线程。接下来我们在 Linux 环境下看一看 NodeJS 的多个线程。

查看 NodeJS 多线程

首先,我们需要先编写一个 js 脚本,写入一个定时器使得脚本不会因为执行完毕而被关掉。

setInterval(function () {}, 1000)

node命令执行该脚本,开启另一个窗口(或者把程序放后台执行)来查看 NodeJS 进程下的线程情况。

$ ps -a  PID TTY          TIME CMD16699 pts/2    00:00:00 node16706 pts/0    00:00:00 ps$ ps -L -p 16699  PID   LWP TTY          TIME CMD16699 16699 pts/2    00:00:00 node16699 16700 pts/2    00:00:00 V8 WorkerThread16699 16701 pts/2    00:00:00 V8 WorkerThread16699 16702 pts/2    00:00:00 V8 WorkerThread16699 16703 pts/2    00:00:00 V8 WorkerThread16699 16704 pts/2    00:00:00 node

可以看到包括 V8 引擎的工作线程在内,已经开启了 6 个线程(MAC OS 系统用ps -M -p <pid>命令)。当前,线程池还未被创建,只有进行 I/O 操作后,线程池才会被创建。在脚本中添加异步读取文件的代码来激活线程池。

require('fs').readFile('test.js', function () {})setInterval(function () {}, 1000)

重新启动脚本,可以看到,启动的 4 个新线程正是 libuv 线程池默认的 4 个线程。

$ ps -a  PID TTY          TIME CMD16745 pts/2    00:00:00 node16755 pts/0    00:00:00 ps$ ps -L -p 16745  PID   LWP TTY          TIME CMD16745 16745 pts/2    00:00:00 node16745 16746 pts/2    00:00:00 V8 WorkerThread16745 16747 pts/2    00:00:00 V8 WorkerThread16745 16748 pts/2    00:00:00 V8 WorkerThread16745 16749 pts/2    00:00:00 V8 WorkerThread16745 16750 pts/2    00:00:00 node16745 16751 pts/2    00:00:00 node16745 16752 pts/2    00:00:00 node16745 16753 pts/2    00:00:00 node16745 16754 pts/2    00:00:00 node

可以通过修改环境变量process.env.UV_THREADPOOL_SIZE(最大 128)使 NodeJS 支持更多地线程。

// jsprocess.env.UV_THREADPOOL_SIZE = 64require('fs').readFile('test.js', function () {})setInterval(function () {}, 1000)// bash$ ps -a  PID TTY          TIME CMD16782 pts/2    00:00:00 node16852 pts/0    00:00:00 ps$ ps -L -p 16782 | wc -l71

重新执行脚本,可以看到减去第一行和 6 个初始线程,有 64 个线程在为 NodeJS 的异步 I/O 服务。

高并发和高可用

JavaScript 是单线程,但 JavaScript 引擎能够创建多个线程来服务与主线程,而 NodeJS 的主线程就像一个调度员,它能够将 I/O 操作,例如网络请求,分发给其它线程进行处理,在通过事件机制将结果返回给主线程,因此,NodeJS 编写的服务器能够支持极大的并发量,这也是 NodeJS 的优势所在。NodeJS 主线程不宜进行大量地计算,因为这会阻塞主线程的运行。所以一般来说,NodeJS 适合 I/O 密集型场景,不适合 CPU 密集型场景。

除了多线程的支持,NodeJS 还提供 child_process 和 cluster 接口允许用户创建很多子进程来处理任务。单线程的 NodeJS 应用是脆弱了,但群体的力量是强大的。多进程、多线程的 NodeJS 才是服务器性能和稳定性的保证。

参考资料

转载地址:http://eoxsa.baihongyu.com/

你可能感兴趣的文章
使用HttpWebRequest出错时获取详细的错误信息
查看>>
sql还原(.mdf文件还原)
查看>>
Mellanox infinoband RDMA SDP
查看>>
Nearest Common Ancestors(LCA)
查看>>
JS/atan2 pow
查看>>
Pythoh网络编程3:创建TCP服务器和客户端
查看>>
angularjs 出现 “Possibly unhandled rejection: cancel ”错误
查看>>
bzoj 2653 middle (主席树+二分)
查看>>
指导别人,弥补自己
查看>>
BZOJ3932: [CQOI2015]任务查询系统
查看>>
和make相关的一些命令
查看>>
Fiddler抓取https设置及其原理
查看>>
常用的一些模板
查看>>
WPF使用Expression Design设计图形
查看>>
Ubuntu 下Qt安装实用教程
查看>>
DNS 协议2
查看>>
Ubuntu 隐藏所有窗口快捷键不生效问题
查看>>
编译Spring源码
查看>>
javascript运算符优先级
查看>>
Spring Cloud 学习 (七) Spring Cloud Sleuth
查看>>