Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

青灯夜游
Lepaskan: 2022-10-14 20:05:41
ke hadapan
1971 orang telah melayarinya

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

I. 前言

本文论点主要面向 Node.js 开发语言

>> Show Me Code,目前代码正在 dev 分支,已完成单元测试,尚待测试所有场景。

>> 建议通读 Node.js 官方文档 -【不要阻塞事件循环】

Node.js 即服务端 Javascript,得益于宿主环境的不同,它拥有比在浏览器上更多的能力。比如:完整的文件系统访问权限、网络协议、套接字编程、进程和线程操作、C++ 插件源码级的支持、Buffer 二进制、Crypto 加密套件的天然支持。【相关教程推荐:nodejs视频教程

Node.js 的是一门单线程的语言,它基于 V8 引擎开发,v8 在设计之初是在浏览器端对 JavaScript 语言的解析运行引擎,其最大的特点是单线程,这样的设计避免了一些多线程状态同步问题,使得其更轻量化易上手。

一、名词定义

1. 进程

学术上说,进程是一个具有一定独立功能的程序在一个数据集上的一次动态执行的过程,是操作系统进行资源分配和调度的一个独立单位,是应用程序运行的载体。我们这里将进程比喻为工厂的车间,它代表 CPU 所能处理的单个任务。任一时刻,CPU 总是运行一个进程,其他进程处于非运行状态。

进程具有以下特性:

  • 进程是拥有资源的基本单位,资源分配给进程,同一进程的所有线程共享该进程的所有资源;
  • 进程之间可以并发执行;
  • 在创建或撤消进程时,系统都要为之分配和回收资源,与线程相比系统开销较大;
  • 一个进程可以有多个线程,但至少有一个线程;

2. 线程

在早期的操作系统中并没有线程的概念,进程是能拥有资源和独立运行的最小单位,也是程序执行的最小单位。任务调度采用的是时间片轮转的抢占式调度方式,而进程是任务调度的最小单位,每个进程有各自独立的一块内存,使得各个进程之间内存地址相互隔离。

后来,随着计算机的发展,对 CPU 的要求越来越高,进程之间的切换开销较大,已经无法满足越来越复杂的程序的要求了。于是就发明了线程,线程是程序执行中一个单一的顺序控制流程,是程序执行流的最小单元。这里把线程比喻一个车间的工人,即一个车间可以允许由多个工人协同完成一个任务,即一个进程中可能包含多个线程。

线程具有以下特性:

  • 线程作为调度和分配的基本单位;
  • 多个线程之间也可并发执行;
  • 线程是真正用来执行程序的,执行计算的;
  • 线程不拥有系统资源,但可以访问隶属于进程的资源,一个线程只能属于一个进程;

Node.js 的多进程有助于充分利用 CPU 等资源,Node.js 的多线程提升了单进程上任务的并行处理能力。

在 Node.js 中,每个 worker 线程都有他自己的 V8 实例和事件循环机制 (Event Loop)。但是,和进程不同,workers 之间是可以共享内存的。

二、Node.js 异步机制

1. Node.js 内部线程池、异步机制以及宏任务优先级划分

Node.js 的单线程是指程序的主要执行线程是单线程,这个主线程同时也负责事件循环。而其实语言内部也会创建线程池来处理主线程程序的 网络 IO / 文件 IO / 定时器 等调用产生的异步任务。一个例子就是定时器 Timer 的实现:在 Node.js 中使用定时器时,Node.js 会开启一个定时器线程进行计时,计时结束时,定时器回调函数会被放入位于主线程的宏任务队列。当事件循环系统执行完主线程同步代码和当前阶段的所有微任务时,该回调任务最后再被取出执行。所以 Node.js 的定时器其实是不准确的,只能保证在预计时间时我们的回调任务被放入队列等待执行,而不是直接被执行。

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

Mekanisme berbilang benang yang digabungkan dengan sistem gelung peristiwa Node.js membolehkan pembangun menggunakan mekanisme tak segerak, termasuk pemasa, IO dan permintaan rangkaian, dalam satu urutan. Walau bagaimanapun, untuk mencapai pelayan yang sangat responsif dan berprestasi tinggi, Gelung Acara Node.js lebih mengutamakan tugas makro.

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

Keutamaan antara tugas makro Node.js: Pemasa >

  • Panggil Balik Pemasa: Apabila tiba masanya, semakin awal pelaksanaan, semakin tepat ia, jadi ini mempunyai keutamaan tertinggi dan mudah difahami.
  • Panggil Balik Belum Selesai: Panggilan Balik semasa mengendalikan rangkaian, IO dan pengecualian lain Sesetengah sistem unix akan menunggu ralat dilaporkan, jadi ia mesti dikendalikan.
  • Panggil Balik Tinjauan: Memproses data IO dan sambungan rangkaian Inilah yang paling dikendalikan oleh pelayan.
  • Semak Panggilan Balik: Panggilan balik untuk melaksanakan setImmediate Cirinya ialah ini boleh dipanggil semula sejurus selepas IO dilaksanakan.
  • Tutup Panggilan Balik: Panggilan balik untuk menutup sumber Perlaksanaan yang tertunda tidak akan menjejaskannya dan mempunyai keutamaan yang paling rendah.

Pengoptimuman dan pembahagian antara tugas mikro Node.js: process.nextTick >

2. Masa pelaksanaan macrotasks dan microtasks Node.js

Sebelum nod 11, Node.js’ Event Loop tidak dilaksanakan satu demi satu seperti penyemak imbas. Tugas makro, dan kemudian laksanakan semua tugas mikro, tetapi laksanakan sebilangan tugas makro Pemasa tertentu, kemudian laksanakan semua tugas mikro, kemudian laksanakan sebilangan tugas makro Menunggu tertentu, dan kemudian laksanakan semua tugas mikro, dan Tinjauan yang selebihnya Begitu juga. benar untuk tugas makro , Semak dan Tutup. Selepas nod 11, ia telah diubah bahawa setiap tugas makro melaksanakan semua tugas mikro.

Tugas makro Node.js juga mempunyai keutamaan Jika Gelung Peristiwa Node.js menjalankan semua tugasan makro keutamaan semasa setiap kali, ia akan menjalankan tugasan Makro seterusnya berlakunya keadaan "kebuluran". Jika terdapat terlalu banyak tugasan makro dalam peringkat tertentu, peringkat seterusnya tidak akan dilaksanakan Oleh itu, setiap jenis tugasan makro mempunyai mekanisme untuk mengehadkan bilangan pelaksanaan, dan selebihnya akan diserahkan kepada Gelung Acara seterusnya untuk pelaksanaan berterusan. .

Prestasi akhir ialah: iaitu, melaksanakan sebilangan tugas makro Pemasa, melaksanakan semua tugas mikro antara setiap tugas makro, dan kemudian melaksanakan sebilangan tugas makro Panggilan Balik Tertunda dan melaksanakan semua tugas mikro antara setiap tugasan makro.

3. Pelbagai proses Node.js

1. Gunakan kaedah child_process untuk mencipta proses secara manual

Program Node.js menyediakan keupayaan untuk melahirkan proses anak melalui modul child_process menyediakan pelbagai cara untuk mencipta proses anak:

  • melahirkan. mencipta proses baharu, dan hasil pelaksanaan distrim Pulangan dalam bentuk, data hasil hanya boleh diperoleh melalui peristiwa, yang menyusahkan untuk dikendalikan.
const spawn = require('child_process').spawn;
const ls = spawn('ls', ['-lh', '/usr']);

ls.stdout.on('data', (data) => {
  console.log(`stdout: ${data}`);
});

ls.stderr.on('data', (data) => {
  console.log(`stderr: ${data}`);
});

ls.on('close', (code) => {
  console.log(`child process exited with code ${code}`);
});
Salin selepas log masuk
  • execFile Cipta proses baharu dan laksanakan fail boleh laku mengikut nama Fail di belakangnya Anda boleh membawa pilihan untuk mengembalikan hasil panggilan dalam bentuk panggilan balik Anda boleh mendapatkan data yang lengkap, yang lebih mudah.
execFile('/path/to/node', ['--version'], function(error, stdout, stderr){
    if(error){
        throw error;
    }
    console.log(stdout);
});
Salin selepas log masuk
  • exec mencipta proses baharu, yang boleh secara langsung melaksanakan arahan shell, memudahkan pelaksanaan arahan shell dan hasil pelaksanaan dikembalikan dalam bentuk panggilan balik.
exec('ls -al', function(error, stdout, stderr){
    if(error) {
        console.error('error:' + error);
        return;
    }
    console.log('stdout:' + stdout);
    console.log('stderr:' + typeof stderr);
});
Salin selepas log masuk
  • fork Buat proses baharu dan laksanakan program nod Proses ini mempunyai contoh V8 yang lengkap, komunikasi IPC dari proses utama proses anak didayakan secara automatik.
var child = child_process.fork('./anotherSilentChild.js', {
    silent: true
});

child.stdout.setEncoding('utf8');
child.stdout.on('data', function(data){
    console.log(data);
});
Salin selepas log masuk
Antaranya, spawn adalah asas kepada semua kaedah, dan lapisan bawah exec memanggil execFile.

2 Gunakan kaedah kluster untuk mencipta proses secara separa automatik

Berikut ialah contoh mudah menggunakan modul

untuk mencipta kluster perkhidmatan http. . Dalam contoh, fail pelaksanaan Js yang sama digunakan semasa membuat Kluster Gunakan Cluster dalam fail untuk menentukan sama ada persekitaran pelaksanaan semasa berada dalam proses utama atau proses anak Jika ia adalah proses utama, gunakan pelaksanaan semasa fail untuk mencipta contoh proses kanak-kanak Jika ia adalah proses kanak-kanak, kemudian Masukkan aliran pemprosesan perniagaan proses anak. Modul cluster.isPrimary

/*
  简单示例:使用同一个 JS 执行文件创建子进程集群 Cluster
*/
const cluster = require('node:cluster');
const http = require('node:http');
const numCPUs = require('node:os').cpus().length;
const process = require('node:process');

if (cluster.isPrimary) {
  console.log(`Primary ${process.pid} is running`);
  // Fork workers.
  for (let i = 0; i  {
    console.log(`worker ${worker.process.pid} died`);
  });
} else {
  // Workers can share any TCP connection
  http.createServer((req, res) => {
    res.writeHead(200);
    res.end('hello world\n');
  }).listen(8000);
  console.log(`Worker ${process.pid} started`);
}
Salin selepas log masuk

membenarkan penubuhan proses utama dan beberapa sub-proses Gunakan Cluster untuk secara tersirat mencipta sub-proses secara dalaman, dan proses utama memantau dan menyelaraskan perjalanan. sub-proses. child_process.fork()

Komunikasi antara proses digunakan untuk bertukar-tukar mesej antara sub-proses Modul Kluster mempunyai pengimbang beban terbina dalam dan menggunakan algoritma Round-robin (pelaksanaan seterusnya) untuk menyelaraskan beban antara sub-proses. . Apabila berjalan, semua sambungan yang baru ditubuhkan diselesaikan oleh proses utama, dan kemudian proses utama memperuntukkan sambungan TCP kepada proses anak yang ditentukan.

Proses kanak-kanak yang dibuat menggunakan kluster boleh menggunakan port yang sama Node.js mempunyai sokongan khas untuk

modul terbina dalam. Proses utama Node.js bertanggungjawab untuk mendengar port sasaran dan selepas menerima permintaan, mengedarkan permintaan kepada proses kanak-kanak mengikut dasar pengimbangan beban. http/net

3. 使用基于 Cluster 封装的 PM2 工具全自动创建进程

PM2 是常用的 node 进程管理工具,它可以提供 node.js 应用管理能力,如自动重载、性能监控、负载均衡等。

其主要用于 独立应用 的进程化管理,在 Node.js 单机服务部署方面比较适合。可以用于生产环境下启动同个应用的多个实例提高 CPU 利用率、抗风险、热加载等能力。

由于是外部库,需要使用 npm 包管理器安装:

$: npm install -g pm2
Salin selepas log masuk

pm2 支持直接运行 server.js 启动项目,如下:

$: pm2 start server.js
Salin selepas log masuk

即可启动 Node.js 应用,成功后会看到打印的信息:

┌──────────┬────┬─────────┬──────┬───────┬────────┬─────────┬────────┬─────┬───────────┬───────┬──────────┐
│ App name │ id │ version │ mode │ pid   │ status │ restart │ uptime │ cpu │ mem       │ user  │ watching │
├──────────┼────┼─────────┼──────┼───────┼────────┼─────────┼────────┼─────┼───────────┼───────┼──────────┤
│ server   │ 0  │ 1.0.0   │ fork │ 24776 │ online │ 9       │ 19m    │ 0%  │ 35.4 MB   │ 23101 │ disabled │
└──────────┴────┴─────────┴──────┴───────┴────────┴─────────┴────────┴─────┴───────────┴───────┴──────────┘
Salin selepas log masuk

pm2 也支持配置文件启动,通过配置文件 ecosystem.config.js 可以定制 pm2 的各项参数:

module.exports = {
  apps : [{
    name: 'API', // 应用名
    script: 'app.js', // 启动脚本
    args: 'one two', // 命令行参数
    instances: 1, // 启动实例数量
    autorestart: true, // 自动重启
    watch: false, // 文件更改监听器
    max_memory_restart: '1G', // 最大内存使用亮
    env: { // development 默认环境变量
      // pm2 start ecosystem.config.js --watch --env development
      NODE_ENV: 'development'
    },
    env_production: { // production 自定义环境变量
      NODE_ENV: 'production'
    }
  }],

  deploy : {
    production : {
      user : 'node',
      host : '212.83.163.1',
      ref  : 'origin/master',
      repo : 'git@github.com:repo.git',
      path : '/var/www/production',
      'post-deploy' : 'npm install && pm2 reload ecosystem.config.js --env production'
    }
  }
};
Salin selepas log masuk

pm2 logs 日志功能也十分强大:

$: pm2 logs
Salin selepas log masuk

II. Node.js 中进程池和线程池的适用场景

一般我们使用计算机执行的任务包含以下几种类型的任务:

  • 计算密集型任务:任务包含大量计算,CPU 占用率高。

    const matrix = {};
    for (let i = 0; i 
    Salin selepas log masuk
  • IO 密集型任务:任务包含频繁的、持续的网络 IO 和磁盘 IO 的调用。

    const {copyFileSync, constants} = require('fs');
    copyFileSync('big-file.zip', 'destination.zip');
    Salin selepas log masuk
  • 混合型任务:既有计算也有 IO。

一、进程池的适用场景

使用进程池的最大意义在于充分利用多核 CPU 资源,同时减少子进程创建和销毁的资源消耗

进程是操作系统分配资源的基本单位,使用多进程架构能够更多的获取 CPU 时间、内存等资源。为了应对 CPU-Sensitive 场景,以及充分发挥 CPU 多核性能,Node 提供了 child_process 模块用于创建子进程。

子进程的创建和销毁需要较大的资源成本,因此池化子进程的创建和销毁过程,利用进程池来管理所有子进程。

除了这一点,Node.js 中子进程也是唯一的执行二进制文件的方式,Node.js 可通过流 (stdin/stdout/stderr) 或 IPC 和子进程通信。

通过 Stream 通信

const {spawn} = require('child_process');
const ls = spawn('ls', ['-lh', '/usr']);

ls.stdout.on('data', (data) => {
  console.log(`stdout: ${data}`);
});

ls.stderr.on('data', (data) => {
  console.error(`stderr: ${data}`);
});

ls.on('close', (code) => {
  console.log(`child process exited with code ${code}`);
});
Salin selepas log masuk

通过 IPC 通信

const cp = require('child_process');
const n = cp.fork(`${__dirname}/sub.js`);

n.on('message', (m) => {
  console.log('PARENT got message:', m);
});

n.send({hello: 'world'});
Salin selepas log masuk

二、线程池的适用场景

使用线程池的最大意义在于多任务并行,为主线程降压,同时减少线程创建和销毁的资源消耗。单个 CPU 密集性的计算任务使用线程执行并不会更快,甚至线程的创建、销毁、上下文切换、线程通信、数据序列化等操作还会额外增加资源消耗。

但是如果一个计算机程序中有很多同一类型的阻塞任务需要执行,那么将他们交给线程池可以成倍的减少任务总的执行时间,因为在同一时刻多个线程在并行进行计算。如果多个任务只使用主线程执行,那么最终消耗的时间是线性叠加的,同时主线程阻塞之后也会影响其它任务的处理。

特别是对 Node.js 这种单主线程的语言来讲,主线程如果消耗了过多的时间来执行这些耗时任务,那么对整个 Node.js 单个进程实例的性能影响将是致命的。这些占用着 CPU 时间的操作将导致其它任务获取的 CPU 时间不足或 CPU 响应不够及时,被影响的任务将进入 “饥饿” 状态。

因此 Node.js 启动后主线程应尽量承担调度的角色,批量重型 CPU 占用任务的执行应交由额外的工作线程处理,主线程最后拿到工作线程的执行结果再返回给任务调用方。另一方面由于 IO 操作 Node.js 内部作了优化和支持,因此 IO 操作应该直接交给主线程,主线程再使用内部线程池处理。

Node.js 的异步能不能解决过多占用 CPU 任务的执行问题?

答案是:不能,过多的异步 CPU 占用任务会阻塞事件循环。

Node.js 的异步在 网络 IO / 磁盘 IO 处理上很有用,宏任务微任务系统 + 内部线程调用能分担主进程的执行压力。但是如果单独将 CPU 占用任务放入宏任务队列或微任务队列,对任务的执行速度提升没有任何帮助,只是一种任务调度方式的优化而已。

我们只是延迟了任务的执行或是将巨大任务分散成多个再分批执行,但是任务最终还是要在主线程被执行。如果这类任务过多,那么任务分片和延迟的效果将完全消失,一个任务可以,那十个一百个呢?量变将会引起质变。

以下是 Node.js 官方博客中的原文:

“如果你需要做更复杂的任务,拆分可能也不是一个好选项。这是因为拆分之后任务仍然在事件循环线程中执行,并且你无法利用机器的多核硬件能力。 请记住,事件循环线程只负责协调客户端的请求,而不是独自执行完所有任务。 对一个复杂的任务,最好把它从事件循环线程转移到工作线程池上。”

  • 场景:间歇性让主进程 瘫痪

每一秒钟,主线程有一半时间被占用

// this task costs 100ms
function doHeavyTask() { ...}

setInterval(() => {
  doHeavyTask(); // 100ms
  doHeavyTask(); // 200ms
  doHeavyTask(); // 300ms
  doHeavyTask(); // 400ms
  doHeavyTask(); // 500ms
}, 1e3);
Salin selepas log masuk
  • 场景:高频性让主进程 半瘫痪

每 200ms,主线程有一半时间被占用

// this task costs 100ms
function doHeavyTask() { ...}

setInterval(() => {
  doHeavyTask();
}, 1e3);

setInterval(() => {
  doHeavyTask();
}, 1.2e3);

setInterval(() => {
  doHeavyTask();
}, 1.4e3);

setInterval(() => {
  doHeavyTask();
}, 1.6e3);

setInterval(() => {
  doHeavyTask();
}, 1.8e3);
Salin selepas log masuk

以下是官方博客的原文摘录:

“因此,你应该保证永远不要阻塞事件轮询线程。换句话说,每个 JavaScript 回调应该快速完成。这些当然对于 await,Promise.then 也同样适用。”

III. 进程池

进程池是对进程的创建、执行任务、销毁等流程进行管控的一个应用或是一套程序逻辑。之所以称之为池是因为其内部包含多个进程实例,进程实例随时都在进程池内进行着状态流转,多个创建的实例可以被重复利用,而不是每次执行完一系列任务后就被销毁。因此,进程池的部分存在目的是为了减少进程创建的资源消耗。

此外进程池最重要的一个作用就是负责将任务分发给各个进程执行,各个进程的任务执行优先级取决于进程池上的负载均衡运算,由算法决定应该将当前任务派发给哪个进程,以达到最高的 CPU 和内存利用率。常见的负载均衡算法有:

  • POLLING - 轮询:子进程轮流处理请求
  • WEIGHTS - 权重:子进程根据设置的权重来处理请求
  • RANDOM - 随机:子进程随机处理请求
  • SPECIFY - 指定:子进程根据指定的进程 id 处理请求
  • WEIGHTS_POLLING - 权重轮询:权重轮询策略与轮询策略类似,但是权重轮询策略会根据权重来计算子进程的轮询次数,从而稳定每个子进程的平均处理请求数量。
  • WEIGHTS_RANDOM - 权重随机:权重随机策略与随机策略类似,但是权重随机策略会根据权重来计算子进程的随机次数,从而稳定每个子进程的平均处理请求数量。
  • MINIMUM_CONNECTION - 最小连接数:选择子进程上具有最小连接活动数量的子进程处理请求。
  • WEIGHTS_MINIMUM_CONNECTION - 权重最小连接数:权重最小连接数策略与最小连接数策略类似,不过各个子进程被选中的概率由连接数和权重共同决定。

一、要点

「 对单一任务的控制不重要,对单个进程宏观的资源占用更需关注 」

二、流程设计

进程池架构图参考之前的进程管理工具开发相关 文章,本文只需关注进程池部分。

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

1. 关键流程

  • 进程池创建进程时会初始化进程实例内的 ProcessHost 事务对象,进程实例向事务对象注册多种任务监听器。
  • 用户向进程池发起单个任务调用请求,可传入进程绑定的 ID 和指定的任务名。
  • 判断用户是否传入 ID 参数指定使用某个进程执行任务,如果未指定 ID:
    • 进程池判断当前进程池进程数量是否已超过最大值,如果未超过则创建新进程,用此进程处理当前任务,并将进程放入进程池。
    • 如果进程池进程数量已达最大值,则根据负载均衡算法选择一个进程处理当前任务。
  • 指定 ID 时:
    • 通过用户传入的 ID 参数找到对应进程,将任务分发给此进程执行。
    • 如果未找到 ID 所对应的进程,则向用户抛出异常。
  • 任务由进程池派发给目标进程后,ProcessHost 事务对象会根据该任务的任务名触发子进程内的监听器。
  • 子进程内的监听器函数可执行同步任务和异步任务,异步任务返回 Promise 对象,同步任务返回值。
  • ProcessHost 事务对象的监听器函数执行完毕后,会将任务结果返回给进程池,进程池再将结果通过异步回调函数返回给用户。
  • 用户也可向进程池所有子进程发起个任务调用请求,最终将会通过 Promise 的返回所有子进程的任务执行结果。

2. 名词解释

  • ProcessHost 事务中心:运行在子进程中,用于事件触发以及和主进程通信。开发者在子进程执行文件中向其注册多个具有特定任务名的任务事件,主进程会向某个子进程发送任务请求,并由事务中心调用指定的事件监听器处理请求。
  • LoadBalancer 负载均衡器:用于选择一个进程处理任务,可根据不同的负载均衡算法实现不同的选择策略。
  • LifeCycle: 设计之初用于管控子进程的智能启停,某个进程在长时间未被使用时进入休眠状态,当有新任务到来时再唤醒进程。目前还有些难点需要解决,比如进程的唤醒和休眠不好实现,进程的使用情况不好统计,该功能暂时不可用。

三、进程池使用方式

更多示例见:进程池 mocha 单元测试

1. 创建进程池

main.js

const { ChildProcessPool, LoadBalancer } = require('electron-re');

const processPool = new ChildProcessPool({
  path: path.join(__dirname, 'child_process/child.js'),
  max: 4,
  strategy: LoadBalancer.ALGORITHM.POLLING,
);
Salin selepas log masuk

child.js

const { ProcessHost } = require('electron-re');

ProcessHost
  .registry('test1', (params) => {
    console.log('test1');
    return 1 + 1;
  })
  .registry('test2', (params) => {
    console.log('test2');
    return new Promise((resolve) => resolve(true));
  });
Salin selepas log masuk

2. 向一个子进程发送任务请求

processPool.send('test1', { value: "test1"}).then((result) => {
  console.log(result);
});
Salin selepas log masuk

3. 向所有子进程发送任务请求

processPool.sendToAll('test1', { value: "test1"}).then((results) => {
  console.log(results);
});
Salin selepas log masuk

四、进程池实际使用场景

1. Electron 网页代理工具中多进程的应用

1)基本代理原理:

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

2)单进程下客户端执行原理:

  • 通过用户预先保存的服务器配置信息,使用 node.js 子进程来启动 ss-local 可执行文件建立和 ss 服务器的连接来代理用户本地电脑的流量,每个子进程占用一个 socket 端口。
  • 其它支持 socks5 代理的 proxy 工具比如:浏览器上的 SwitchOmega 插件会和这个端口的 tcp 服务建立连接,将 tcp 流量加密后通过代理服务器转发给我们需要访问的目标服务器。

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

3)多进程下客户端执行原理:

以上描述的是客户端连接单个节点的工作模式,节点订阅组中的负载均衡模式需要同时启动多个子进程,每个子进程启动 ss-local 执行文件占用一个本地端口并连接到远端一个服务器节点。

每个子进程启动时选择的端口是会变化的,因为某些端口可能已经被系统占用,程序需要先选择未被使用的端口。并且浏览器 proxy 工具也不可能同时连接到我们本地启动的子进程上的多个 ss-local 服务上。因此需要一个占用固定端口的中间节点接收 proxy 工具发出的连接请求,然后按照某种分发规则将 tcp 流量转发到各个子进程的 ss-local 服务的端口上。

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

2. Muat naik tersegmen fail berbilang proses Klien elektron

Sebelum ini saya telah membuat klien yang menyokong muat naik berbilang fail bersegmen protokol SMB, pengurusan tugas muat naik sisi Node.js, IO operasi, dsb. semuanya telah dilaksanakan menggunakan pelbagai proses dalam satu versi, tetapi saya melakukannya sendiri (melarikan diri) dalam cawangan eksperimen gitlab.

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

IV. Kolam Benang

Untuk mengurangkan overhed sistem pengiraan tugas intensif CPU, Node.js telah memperkenalkan Ciri baharu: worker_threads, yang pertama kali muncul sebagai ciri percubaan dalam v10.5.0. Berbilang utas boleh dibuat dalam proses melalui worker_threads Utas utama dan utas pekerja berkomunikasi menggunakan parentPort, dan utas pekerja boleh berkomunikasi secara langsung melalui MessageChannel. Sebagai ciri penting untuk pembangun menggunakan benang, worker_threads boleh digunakan secara normal dalam persekitaran pengeluaran dalam versi v12.11.0 stabil.

Walau bagaimanapun, penciptaan utas memerlukan sumber CPU dan memori tambahan Jika utas hendak digunakan beberapa kali, ia harus disimpan Apabila utas tidak digunakan sama sekali, ia perlu ditutup tepat pada masanya untuk mengurangkan penggunaan memori. Bayangkan bahawa kita mencipta benang secara langsung apabila kita perlu menggunakannya, dan memusnahkannya serta-merta selepas digunakan Mungkin kos mencipta dan memusnahkan benang itu sendiri telah melebihi kos sumber yang disimpan dengan menggunakan benang itu sendiri. Walaupun Node.js menggunakan kumpulan benang secara dalaman, ia benar-benar telus dan tidak dapat dilihat oleh pembangun Oleh itu, kepentingan merangkum alat kumpulan benang yang boleh mengekalkan kitaran hayat benang ditunjukkan.

Untuk meningkatkan penjadualan berbilang tugas tak segerak, kumpulan benang bukan sahaja menyediakan keupayaan untuk mengekalkan urutan, tetapi juga menyediakan keupayaan untuk mengekalkan baris gilir tugas. Apabila menghantar permintaan kepada kumpulan benang untuk melaksanakan tugasan tak segerak, jika tiada benang terbiar dalam kumpulan benang, tugas itu akan dibuang terus Jelas sekali ini bukan kesan yang diingini.

Oleh itu, anda boleh mempertimbangkan untuk menambah logik penjadualan baris gilir tugas pada kumpulan benang: apabila kumpulan benang tidak mempunyai benang terbiar, masukkan tugasan itu ke dalam baris gilir tugasan untuk dilaksanakan (FIFO), dan kumpulan benang akan mengambil tugas pada masa tertentu Ia dilaksanakan oleh utas terbiar Selepas pelaksanaan selesai, fungsi panggil balik tak segerak dicetuskan dan hasil pelaksanaan dikembalikan kepada pemanggil permintaan. Walau bagaimanapun, bilangan tugasan dalam baris gilir tugas kumpulan benang hendaklah dihadkan kepada nilai khas untuk mengelakkan beban berlebihan pada kumpulan benang daripada menjejaskan prestasi keseluruhan aplikasi Node.js.

1. Perkara utama

"Adalah penting untuk mengawal satu tugas, dan tidak perlu memberi perhatian kepada pekerjaan sumber daripada satu utas"

2. Reka bentuk terperinci

Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node

Proses aliran tugas

  • Pemanggil boleh menghantar tugas ke kumpulan benang melalui StaticPool/StaticExcutor/DynamicPool/DynamicExcutor kejadian (istilah utama diterangkan di bawah Perbezaan terbesar antara pelbagai kejadian ialah keupayaan dinamik parameter).

  • Tugas dijana secara dalaman oleh kumpulan benang Selepas penjanaan, tugasan digunakan sebagai pembawa edaran utama Di satu pihak, ia membawa parameter pengiraan tugas yang dihantar oleh pengguna , dan sebaliknya, ia merekodkan status proses edaran tugasan, seperti: status tugas, masa mula, masa tamat, ID tugas, bilangan percubaan semula tugas, sama ada tugasan menyokong percubaan semula, jenis tugasan, dsb.

  • Selepas tugasan dijana, mula-mula tentukan sama ada bilangan utas dalam kumpulan utas semasa telah mencapai had atas Jika belum mencapai had atas, buat urutan baharu dan letakkannya ke dalam kawasan penyimpanan benang, dan kemudian gunakan benang Laksanakan tugas semasa secara langsung.

  • Jika bilangan utas dalam kumpulan utas melebihi had, tentukan sama ada terdapat utas melahu yang belum melaksanakan tugasan Selepas mendapat utas melahu, gunakan utas ini untuk melaksanakan secara langsung tugas semasa.

  • Jika tiada urutan terbiar, ia akan dinilai sama ada baris gilir tugasan semasa telah penuh Jika baris gilir tugasan penuh, ralat akan dilemparkan, membolehkan pemanggil merasakan bahawa tugas itu tidak dapat dilaksanakan dengan jayanya buat kali pertama.

  • Jika baris gilir tugasan tidak penuh, masukkan tugasan itu ke dalam baris gilir tugas dan tunggu sistem kitaran tugasan mengeluarkannya dan melaksanakannya.

  • Selepas tugasan dilaksanakan dalam tiga kes langkah 4/5/6 di atas, ia dinilai sama ada tugasan itu berjaya dilaksanakan Apabila berjaya, fungsi panggil balik yang berjaya dicetuskan. dan status Janji diisi. Jika gagal, tentukan sama ada cuba semula disokong Jika cuba semula disokong, tugasan akan dicuba semula 1 dan kemudian diletakkan semula di penghujung baris gilir tugas. Jika tugas itu tidak menyokong cuba semula, ia akan gagal secara langsung dan mencetuskan fungsi panggil balik tak segerak yang gagal, dan status Janji akan ditolak.

  • Dalam keseluruhan kitaran hayat kolam benang, terdapat sistem kitaran tugas, yang memperoleh tugas daripada ketua baris gilir tugas pada frekuensi berkala tertentu, memperoleh benang terbiar daripada storan benang kawasan, dan menggunakan urutan ini untuk melaksanakan Tugas, proses itu juga mematuhi penerangan dalam langkah 7.

  • 任务循环系统除了取任务执行,如果线程池设置了任务超时时间的话,也会判断正在执行中的任务是否超时,超时后会终止该线程的所有运行中的代码。

模块说明

  • StaticPool
    • 定义:静态线程池,可使用固定的 execFunction/execString/execFile 执行参数来启动工作线程,执行参数在进程池创建后不能更改。
    • 进程池创建之后除了执行参数不可变外,其它参数比如:任务超时时间、任务重试次数、线程池任务轮询间隔时间、最大任务数、最大线程数、是否懒创建线程等都可以通过 API 随时更改。
  • StaticExcutor
    • 定义:静态线程池的执行器实例,继承所属线程池的固定执行参数 execFunction/execString/execFile 且不可更改。
    • 执行器实例创建之后除了执行参数不可变外,其它参数比如:任务超时时间、任务重试次数、transferList 等都可以通过 API 随时更改。
    • 静态线程池的各个执行器实例的参数设置互不影响,参数默认继承于所属线程池,参数在执行器上更改后具有比所属线程池同名参数更高的优先级。
  • DynamicPool
    • 定义:动态线程池,无需使用 execFunction/execString/execFile 执行参数即可创建线程池。执行参数在调用 exec() 方法时动态传入,因此执行参数可能不固定。
    • 线程池创建之后执行参数默认为 null,其它参数比如:任务超时时间、任务重试次数、transferList 等都可以通过 API 随时更改。
  • DynamicExcutor
    • 定义:动态线程池的执行器实例,继承所属线程池的其它参数,执行参数为 null
    • 执行器实例创建之后,其它参数比如:任务超时时间、任务重试次数、transferList 等都可以通过 API 随时更改。
    • 动态线程池的各个执行器实例的参数设置互不影响,参数默认继承于所属线程池,参数在执行器上更改后具有比所属线程池同名参数更高的优先级。
    • 动态执行器实例在执行任务之前需要先设置执行参数 execFunction/execString/execFile,执行参数可以随时改变。
  • ThreadGenerator
    • 定义:线程创建的工厂方法,会进行参数校验。
  • Thread
    • 定义:线程实例,内部简单封装了 worker_threads API。
  • TaskGenerator
    • 定义:任务创建的工厂方法,会进行参数校验。
  • Task
    • 定义:单个任务,记录了任务执行状态、任务开始结束时间、任务重试次数、任务携带参数等。
  • TaskQueue
    • 定义:任务队列,在数组中存放任务,以先入先出方式 (FIFO) 向线程池提供任务,使用 Map 来存储 taskId 和 task 之间的映射关系。
  • Task Loop
    • 任务循环,每个循环的默认时间间隔为 2S,每次循环中会处理超时任务、将新任务派发给空闲线程等。

三、线程池使用方式

更多示例见:线程池 mocha 单元测试

1. 创建静态线程池

main.js

const { StaticThreadPool } = require(`electron-re`);
const threadPool = new StaticThreadPool({
  execPath: path.join(__dirname, './worker_threads/worker.js'),
  lazyLoad: true, // 懒加载
  maxThreads: 24, // 最大线程数
  maxTasks: 48, // 最大任务数
  taskRetry: 1, // 任务重试次数
  taskLoopTime: 1e3, // 任务轮询时间
});
const executor = threadPool.createExecutor();
Salin selepas log masuk

worker.js

const fibonaccis = (n) => {
  if (n  {
  return fibonaccis(value);
}
Salin selepas log masuk

2. 使用静态线程池发送任务请求

threadPool.exec(15).then((res) => {
  console.log(+res.data === 610)
});

executor
  .setTaskRetry(2) // 不影响 pool 的全局设置
  .setTaskTimeout(2e3) // 不影响 pool 的全局设置
  .exec(15).then((res) => {
    console.log(+res.data === 610)
  });
Salin selepas log masuk

3. 动态线程池和动态执行器

const { DynamicThreadPool } = require(`electron-re`);
const threadPool = new DynamicThreadPool({
  maxThreads: 24, // 最大线程数
  maxTasks: 48, // 最大任务数
  taskRetry: 1, // 任务重试次数
});
const executor = threadPool.createExecutor({
  execFunction: (value) => { return 'dynamic:' + value; },
});

threadPool.exec('test', {
  execString: `module.exports = (value) => { return 'dynamic:' + value; };`,
});
executor.exec('test');
executor
  .setExecPath('/path/to/exec-file.js')
  .exec('test');
Salin selepas log masuk

四、线程池实际使用场景

暂未在项目中实际使用,可考虑在前端图片像素处理、音视频转码处理等 CPU 密集性任务中进行实践。

这里有篇文章写了 web_worker 的一些应用场景,web_worker 和 worker_threads 是类似的,宿主环境不同,一些权限和能力的不同而已。

V. Ending

Pada mulanya, projek ialah set alat untuk pembangunan aplikasi Elektron yang menyediakan BrowserService / ChildProcessPool / 简易进程监控 UI / 进程间通信 dan fungsi lain kolam benang sebenarnya Ia tidak dirancang pada mulanya, dan kolam benang itu sendiri adalah bebas dan tidak bergantung pada fungsi modul lain dalam elektron-semula Ia sepatutnya bebas pada masa hadapan.

Pelaksanaan kumpulan proses dan kumpulan benang perlu dipertingkatkan.

Sebagai contoh, kumpulan proses tidak menyokong keluar automatik apabila proses anak melahu untuk melepaskan pekerjaan sumber Pada masa itu, versi lain telah dibuat untuk memantau status pelaksanaan tugas ProcessHost untuk membiarkan kanak-kanak memproses tidur apabila terbiar saya ingin menyelamatkan pekerjaan sumber dengan cara ini. Walau bagaimanapun, memandangkan tiada sokongan tahap API node.js untuk membezakan keadaan terbiar proses kanak-kanak, dan fungsi tidur/bangun bagi proses kanak-kanak itu agak tidak berguna (terdapat percubaan untuk mencapainya dengan menghantar SIGSTOP/SIGCONT isyarat kepada proses kanak-kanak), ciri ini akhirnya dimansuhkan.

Kami boleh mempertimbangkan untuk menyokong algoritma pengimbangan beban CPU/Memori pada masa hadapan, kami telah melaksanakan pengumpulan penghunian sumber melalui modul ProcessManager dalam projek.

Ketersediaan relatif kumpulan benang masih tinggi Ia menyediakan pool/excutor dua peringkat pengurusan panggilan, menyokong panggilan berantai dan menyokong kaedah transferList untuk mengelakkan data dalam beberapa senario yang memerlukan prestasi penghantaran data. untuk dipertingkatkan. Berbanding dengan penyelesaian kumpulan benang Node sumber terbuka lain, ia memfokuskan pada mengukuhkan fungsi baris gilir tugas dan menyokong fungsi seperti percubaan semula tugas dan tamat masa tugas.

Untuk lebih banyak pengetahuan berkaitan nod, sila lawati: tutorial nodejs!

Atas ialah kandungan terperinci Mari kita bincangkan tentang cara melaksanakan kumpulan proses ringan dan kumpulan benang menggunakan Node. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

Label berkaitan:
sumber:juejin.cn
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan
Tentang kita Penafian Sitemap
Laman web PHP Cina:Latihan PHP dalam talian kebajikan awam,Bantu pelajar PHP berkembang dengan cepat!