Rumah >hujung hadapan web >tutorial js >Artikel untuk membincangkan model berbilang proses Node dan penggunaan projek
nodBagaimana untuk melaksanakan pelbagai proses? Bagaimana untuk menggunakan projek nod? Artikel berikut akan membantu anda menguasai pengetahuan berkaitan model berbilang proses dan penggunaan projek Node.js, saya harap ia akan membantu anda.
Semalam, seorang rakan bertanya bagaimana untuk menggunakan projek ekspres. Jadi saya menyusun artikel ini, yang terutamanya bercakap tentang cara menggunakan program pelayan yang dibangunkan berdasarkan nodejs, untuk rujukan rakan-rakan yang memerlukan.
Artikel mengandungi beberapa bahagian:
Proses ialah unit asas untuk sistem pengendalian komputer untuk memperuntukkan dan menjadualkan tugas. Buka pengurus tugas dan anda dapat melihat bahawa sebenarnya terdapat banyak program yang berjalan di latar belakang komputer, dan setiap program adalah satu proses. [Tutorial berkaitan yang disyorkan: tutorial video nodejs, Pengajaran pengaturcaraan]
Pelayar moden pada asasnya adalah seni bina berbilang proses Pelayar Chrome sebagai contoh, buka "More Tools" - "Task Manager" dan anda boleh melihat maklumat proses pelayar semasa Salah satu halaman adalah proses Selain itu, terdapat juga proses rangkaian, Proses GPU dll.
Seni bina berbilang proses memastikan operasi aplikasi yang lebih stabil. Mengambil penyemak imbas sebagai contoh, jika semua program berjalan dalam satu proses, jika terdapat kegagalan rangkaian atau ralat pemaparan halaman, ia akan menyebabkan keseluruhan pelayar ranap. Melalui seni bina berbilang proses, walaupun proses rangkaian ranap, ia tidak akan menjejaskan paparan halaman sedia ada, dan paling teruk, ia akan tidak dapat mengakses rangkaian buat sementara waktu.
Benang ialah unit terkecil yang sistem pengendalian boleh melaksanakan penjadualan pengkomputeran. Ia termasuk dalam proses dan merupakan unit operasi sebenar dalam proses. Sebagai contoh, program adalah seperti sebuah syarikat dengan pelbagai jabatan, yang merupakan beberapa proses kerjasama setiap jabatan membolehkan syarikat itu berjalan secara normal, dan benangnya ialah pekerja, orang yang melakukan kerja tertentu.
Kita semua tahu bahawa JavaScript ialah bahasa satu benang. Reka bentuk ini adalah kerana pada zaman awal, JS digunakan terutamanya untuk menulis skrip dan bertanggungjawab untuk merealisasikan kesan interaktif halaman. Jika ia direka bentuk sebagai bahasa berbilang benang, pertama, ia tidak perlu, dan kedua, berbilang benang secara bersama-sama mengendalikan nod DOM, maka nasihat siapakah yang harus didengari oleh penyemak imbas? Sudah tentu, dengan perkembangan teknologi, JS kini turut menyokong multi-threading, tetapi ia hanya digunakan untuk mengendalikan beberapa logik yang tidak berkaitan dengan operasi dom.
Urut tunggal dan satu proses membawa masalah yang serius Untuk program node.js yang sedang berjalan, sebaik sahaja utas utama ditutup, proses itu juga akan ditutup keseluruhan permohonan akan ditutup. Tambahan pula, kebanyakan komputer moden mempunyai CPU berbilang teras, dengan empat teras dan lapan utas, dan lapan teras dan enam belas utas, yang merupakan peranti yang sangat biasa. Sebagai program satu proses, node.js membazirkan prestasi CPU berbilang teras.
Memandangkan situasi ini, kami memerlukan model berbilang proses yang sesuai untuk mengubah program node.js satu proses kepada seni bina berbilang proses.
Terdapat dua penyelesaian biasa untuk melaksanakan seni bina berbilang proses dalam Node.js, kedua-duanya menggunakan modul asli, iaitu modul child_process
dan modul cluster
.
child_process
ialah modul terbina dalam node.js Anda boleh meneka dari namanya bahawa ia bertanggungjawab untuk perkara yang berkaitan dengan proses anak.
Kami tidak akan menghuraikan penggunaan khusus modul ini Malah, ia hanya mempunyai kira-kira enam atau tujuh kaedah, yang sangat mudah difahami. Kami menggunakan salah satu kaedah fork
untuk menunjukkan cara melaksanakan berbilang proses dan komunikasi antara berbilang proses.
Pertama lihat struktur direktori kes demonstrasi yang disediakan:
Kami mencipta pelayan http menggunakan modul http
Apabila terdapat /sum
Apabila permintaan masuk, proses anak akan dibuat melalui modul child_process
dan proses anak akan dimaklumkan untuk melaksanakan logik pengiraan Pada masa yang sama, proses ibu bapa juga akan mendengar mesej yang dihantar oleh anak proses:
// child_process.jsconst http = require('http')const { fork } = require('child_process')const server = http.createServer((req, res) => { if (req.url == '/sum') { // fork 方法接收一个模块路径,然后开启一个子进程,将模块在子进程中运行 // childProcess 表示创建的子进程 let childProcess = fork('./sum.js') // 发消息给子进程 childProcess.send('子进程开始计算') // 父进程中监听子进程的消息 childProcess.on('message', (data) => { res.end(data + '') }) // 监听子进程的关闭事件 childProcess.on('close', () => { // 子进程正常退出和报错挂掉,都会走到这里 console.log('子进程关闭') childProcess.kill() }) // 监听子进程的错误事件 childProcess.on('error', () => { console.log('子进程报错') childProcess.kill() }) } if (req.url == '/hello') { res.end('hello') } // 模拟父进程报错 if (req.url == '/error') { throw new Error('父进程出错') res.end('hello') } }) server.listen(3000, () => { console.log('Server is running on 3000') })复制代码
sum.js
digunakan untuk mensimulasikan tugasan yang perlu dilakukan oleh proses kanak-kanak. Proses anak mendengar mesej yang dihantar oleh proses induk, memproses tugas pengiraan, dan kemudian menghantar keputusan kepada proses induk:
// sum.jsfunction getSum() { let sum = 0 for (let i = 0; i { console.log('主进程的消息:', data) const result = getSum() // 将计算结果发送给父进程 process.send(result) })复制代码
Buka terminal dan jalankan arahan node 1.child_process
:
Lawati penyemak imbas:
Kemudian simulasikan situasi di mana proses kanak-kanak melaporkan ralat:
// sum.jsfunction getSum() { // ....}// 子进程运行5s后,模拟进程挂掉 setTimeout(() => { throw new Error('报错') }, 1000 * 5) process.on('message', (data) => { // ...})复制代码
Lawati penyemak imbas sekali lagi, 5 saat kemudian Perhatikan konsol:
Proses kanak-kanak telah ditutup, dan kemudian akses url lain: /hello
,
Ia boleh dilihat bahawa proses induk masih boleh mengendalikan permintaan dengan betul, yang bermaksud bahawa proses anak melaporkan ralat dan tidak akan menjejaskan operasi proses induk .
Kemudian mari kita simulasikan senario di mana proses induk melaporkan ralat, ulas ralat simulasi modul sum.js
, kemudian mulakan semula perkhidmatan dan akses penyemak imbas /error
:
Selepas mengetahui bahawa proses induk mati, keseluruhan program node.js terkeluar secara automatik, dan perkhidmatan runtuh sepenuhnya, tidak meninggalkan ruang untuk pemulihan.
Ia boleh dilihat bahawa tidak rumit untuk melaksanakan seni bina berbilang proses node.js melalui kaedah child_process
fork
. Komunikasi antara proses terutamanya melalui kaedah send
dan on
Daripada penamaan ini, kita juga dapat mengetahui bahawa lapisan bawah haruslah model publish-subscribe.
Tetapi terdapat masalah serius Walaupun proses anak tidak menjejaskan proses ibu bapa, Setelah proses ibu bapa membuat kesilapan dan menutup telefon, semua proses anak akan "dibunuh dalam satu periuk". . Oleh itu, penyelesaian ini sesuai untuk menggabungkan beberapa operasi yang rumit dan memakan masa ke dalam sub-proses yang berasingan untuk melakukan . Untuk lebih tepat, penggunaan ini digunakan untuk menggantikan pelaksanaan multi-threading, bukan multi-processing.
menggunakan modul child_process
untuk melaksanakan pelbagai pemprosesan, yang nampaknya tidak berguna. Oleh itu, secara amnya disyorkan untuk menggunakan modul cluster
untuk melaksanakan model berbilang proses node.js.
cluster
bermaksud kelompok saya percaya semua orang biasa dengan istilah ini. Sebagai contoh, pada masa lalu, syarikat itu hanya mempunyai satu meja depan, dan kadangkala terlalu sibuk untuk menerima pelawat tepat pada masanya. Kini syarikat telah memperuntukkan empat meja depan Walaupun tiga sibuk, masih ada satu yang boleh menerima pelawat baru. Pengelompokan secara kasar bermaksud ini Untuk perkara yang sama, ia ditetapkan secara munasabah kepada orang yang berbeza untuk melakukannya, untuk memastikan perkara itu dapat dilakukan dengan terbaik.
cluster
Penggunaan modul juga agak mudah. Jika proses semasa adalah proses utama, cipta bilangan proses anak yang sesuai berdasarkan bilangan teras CPU dan dengarkan peristiwa exit
proses anak Jika proses anak keluar, buat semula proses anak baru. Jika ia bukan proses kanak-kanak, perniagaan sebenar diproses.
const http = require('http')const cluster = require('cluster')const cpus = require('os').cpus()if (cluster.isMaster) { // 程序启动时首先走到这里,根据 CPU 的核数,创建出多个子进程 for (let i = 0; i { cluster.fork() }) } else { // fork 方法执行创建子进程,同时会再次执行该模块,此时逻辑就会走到这里 const server = http.createServer((req, res) => { console.log(process.pid) res.end('ok') }) server.listen(3000, () => { console.log('Server is running on 3000', 'pid: ' + process.pid) }) }复制代码
Mulakan perkhidmatan:
Seperti yang anda lihat, modul cluster
telah mencipta banyak proses kanak-kanak, ia nampaknya setiap Anak memproses semuanya menjalankan perkhidmatan web yang sama.
Perlu diingatkan bahawa pada masa ini proses kanak-kanak ini tidak mendengar port yang sama bersama-sama. Pelayan yang dicipta oleh kaedah createServer masih bertanggungjawab untuk pemantauan port dan memajukan permintaan kepada setiap proses anak.
Kami menulis skrip permintaan untuk meminta perkhidmatan di atas dan melihat kesannya.
// request.jsconst http = require('http')for (let i = 0; i <blockquote><p>Modul http bukan sahaja boleh mencipta pelayan http, tetapi juga boleh digunakan untuk menghantar permintaan http. Axios menyokong persekitaran pelayar dan pelayan Di bahagian pelayan, modul http digunakan untuk menghantar permintaan http. </p></blockquote><p>Gunakan perintah <code>node</code> untuk melaksanakan fail dan lihat konsol asal: </p><p style="text-align:center;"><img src="https://img.php.cn/upload/article/000/000/024/6acfb36f7a6b8401b888460a3a451efa-9.png" alt="Artikel untuk membincangkan model berbilang proses Node dan penggunaan projek" style="max-width:90%" loading="lazy"></p><p> mencetak subrutin yang berbeza untuk memproses secara khusus permintaan. ID proses proses. </p><p>Ini ialah seni bina berbilang proses nodd.js yang dilaksanakan melalui modul <code>cluster</code>. </p><p>当然,我们在部署 node.js 项目时不会这么干巴巴的写和使用 <code>cluster</code> 模块。有一个非常好用的工具,叫做 <strong>PM2</strong>,它是一个基于 cluster 模块实现的进程管理工具。在后面的章节中会介绍它的基本用法。</p><h3 data-id="heading-8">小结</h3><p>到此为止,我们花了一部分篇幅介绍 node.js 中多进程的知识,其实仅是想要交代下<strong>为什么需要使用 pm2 来管理 node.js 应用</strong>。本文由于篇幅有限,再加上描述不够准确/详尽,仅做简单介绍。如果是第一次接触这一块内容的朋友,可能没有太明白,也不打紧,后面会再出一篇更细节的文章。</p><h2 data-id="heading-9">部署实践</h2><h3 data-id="heading-10">准备一个 express 项目</h3><p>本文已经准备了一个使用 express 开发的示例程序,<a href="https://www.php.cn/link/7e6bd4cd5be9faae7e6adca908e1995b" target="_blank" rel="nofollow noopener noreferrer" title="https://github.com/hsyq/ndoe-deploy/tree/main/express-demo" ref="nofollow noopener noreferrer">点此访问</a>。</p><p>它主要实现了一个接口服务,当访问 <code>/api/users</code> 时,使用 <code>mockjs</code> 模拟了10条用户数据,返回一个用户列表。同时会开启一个定时器,来模拟报错的情况:</p><pre class="brush:php;toolbar:false">const express = require('express')const Mock = require('mockjs')const app = express() app.get("/api/users", (req, res) => { const userList = Mock.mock({ 'userList|10': [{ 'id|+1': 1, 'name': '@cname', 'email': '@email' }] }) setTimeout(()=> { throw new Error('服务器故障') }, 5000) res.status(200) res.json(userList) }) app.listen(3000, () => { console.log("服务启动: 3000") })复制代码
本地测试一下,在终端中执行命令:
node server.js复制代码
打开浏览器,访问用户列表接口:
五秒钟后,服务器会挂掉:
后面我们使用 pm2 来管理应用后,就可以解决这个问题。
通常完成一个 vue/react 项目后,我们都会先执行打包,再进行发布。其实前端项目要进行打包,主要是因为程序最终的运行环境是浏览器,而浏览器存在各种兼容性问题和性能问题,比如:
.vue
,.jsx
,.ts
文件,需要编译而使用 express.js 或者 koa.js 开发的项目,并不存在这些问题。并且,Node.js 采用 CommonJS 模块化规范,有缓存的机制;同时,只有当模块在被用到时,才会被导入。如果进行打包,打包成一个文件,其实就浪费了这个优势。所以针对 node.js 项目,并不需要打包。
本文以 CentOS 系统为例进行演示。
为了方便切换 node 的版本,我们使用 nvm 来管理 node。
Nvm(Node Version Manager) ,就是 Node.js 的版本管理工具。通过它,可以让 node 在多个版本之间进行任意切换,避免了需要切换版本时反复的下载和安装的操作。
Nvm的官方仓库是 github.com/nvm-sh/nvm。因为它的安装脚本存放在 githubusercontent
站点上,经常访问不了。所以我在 gitee 上新建了它的镜像仓库,这样就能从 gitee 上访问到它的安装脚本了。
通过 curl
命令下载安装脚本,并使用 bash
执行脚本,会自动完成 nvm 的安装工作:
# curl -o- https://gitee.com/hsyq/nvm/raw/master/install.sh | bash复制代码
当安装完成之后,我们再打开一个新的窗口,来使用 nvm :
[root@ecs-221238 ~]# nvm -v0.39.1复制代码
可以正常打印版本号,说明 nvm 已经安装成功了。
现在就可以使用 nvm 来安装和管理 node 了。
查看可用的 node 版本:
# nvm ls-remote复制代码
安装 node:
# nvm install 18.0.0复制代码
查看已经安装的 node 版本:
[root@ecs-221238 ~]# nvm list-> v18.0.0default -> 18.0.0 (-> v18.0.0) iojs -> N/A (default) unstable -> N/A (default) node -> stable (-> v18.0.0) (default) stable -> 18.0 (-> v18.0.0) (default)复制代码
选择一个版本进行使用:
# nvm use 18.0.0复制代码
需要注意的一点,在 Windows 上使用 nvm 时,需要使用管理员权限执行 nvm 命令。在 CentOS 上,我默认使用 root 用户登录的,因而没有出现问题。大家在使用时遇到了未知错误,可以搜索一下解决方案,或者尝试下是否是权限导致的问题。
在安装 node 的时候,会自动安装 npm。查看 node 和 npm 的版本号:
[root@ecs-221238 ~]# node -vv18.0.0[root@ecs-221238 ~]# npm -v8.6.0复制代码
默认的 npm 镜像源是官方地址:
[root@ecs-221238 ~]# npm config get registryhttps://registry.npmjs.org/复制代码
切换为国内淘宝的镜像源:
[root@ecs-221238 ~]# npm config set registry https://registry.npmmirror.com复制代码
到此为止,服务器就已经安装好 node 环境和配置好 npm 了。
方法有很多,或者从 Github / GitLab / Gitee 仓库中下载到服务器中,或者本地通过 ftp 工具上传。步骤很简单,不再演示。
演示项目放到了 /www
目录 下:
一般云服务器仅开放了 22 端口用于远程登录。而常用的80,443等端口并未开放。另外,我们准备好的 express 项目运行在3000端口上。所以需要先到云服务器的控制台中,找到安全组,添加几条规则,开放80和3000端口。
在开发阶段,我们可以使用 nodemon
来做实时监听和自动重启,提高开发效率。在生产环境,就需要祭出大杀器—PM2了。
首先全局安装 pm2:
# npm i -g pm2复制代码
执行 pm2 -v
命令查看是否安装成功:
[root@ecs-221238 ~]# pm2 -v5.2.0复制代码
切换到项目目录,先把依赖装上:
cd /www/express-demo npm install复制代码
然后使用 pm2
命令来启动应用。
pm2 start app.js -i max// 或者pm2 start server.js -i 2复制代码
PM2 管理应用有 fork 和 cluster 两种模式。在启动应用时,通过使用 -i 参数来指定实例的个数,会自动开启 cluster 模式。此时就具备了负载均衡的能力。
-i :instance,实例的个数。可以写具体的数字,也可以配置成 max,
PM2
会自动检查可用的CPU的数量,然后尽可能多地启动进程。
此时应用就启动好了。PM2 会以守护进程的形式管理应用,这个表格展示了应用运行的一些信息,比如运行状态,CPU使用率,内存使用率等。
在本地的浏览器中访问接口:
Cluster 模式是一个多进程多实例的模型,请求进来后会分配给其中一个进程处理。正如前面我们看过的 cluster
模块的用法一样,由于 pm2 的守护,即使某个进程挂掉了,也会立刻重启该进程。
回到服务器终端,执行 pm2 logs
命令,查看下 pm2 的日志:
可见,id 为1的应用实例挂掉了,pm2 会立刻重启该实例。注意,这里的 id 是应用实例的 id,并非进程 id。
到这里,一个 express 项目的简单部署就完成了。通过使用 pm2 工具,基本能保证我们的项目可以稳定可靠的运行。
这里整理了一些 pm2 工具常用的命令,可供查询参考。
# Fork模式pm2 start app.js --name app # 设定应用的名字为 app# Cluster模式# 使用负载均衡启动4个进程pm2 start app.js -i 4 # 将使用负载均衡启动4个进程,具体取决于可用的 CPUpm2 start app.js -i 0 # 等同于上面命令的作用pm2 start app.js -i max # 给 app 扩展额外的3个进程pm2 scale app +3# 将 app 扩展或者收缩到2个进程pm2 scale app 2 # 查看应用状态# 展示所有进程的状态pm2 list # 用原始 JSON 格式打印所有进程列表pm2 jlist# 用美化的 JSON 打印所有进程列表pm2 prettylist # 展示特定进程的所有信息pm2 describe 0# 使用仪表盘监控所有进程pm2 monit # 日志管理# 实时展示所有应用的日志pm2 logs # 实时展示 app 应用的日志 pm2 logs app# 使用json格式实时展示日志,不输出旧日志,只输出新产生的日志pm2 logs --json# 应用管理# 停止所有进程pm2 stop all# 重启所有进程pm2 restart all # 停止指定id的进程pm2 stop 0 # 重启指定id的进程pm2 restart 0 # 删除id为0进程pm2 delete 0# 删除所有的进程pm2 delete all 复制代码
每一条命令都可以亲自尝试一下,看看效果。
这里特别展示下 monit
命令,它可以在终端中启动一个面板,实时展示应用的运行状态,通过上下箭头可以切换 pm2 管理的所有应用:
PM2 的功能十分强大,远不止上面的这几个命令。在真实的项目部署中,可能还需要配置日志文件,watch 模式,环境变量等等。如果每次都手敲命令是十分繁琐的,所以 pm2 提供了配置文件来管理和部署应用。
可以通过以下命令来生成一份配置文件:
[root@ecs-221238 express-demo]# pm2 init simpleFile /www/express-demo/ecosystem.config.js generated复制代码
会生成一个ecosystem.config.js
文件:
module.exports = { apps : [{ name : "app1", script : "./app.js" }] }复制代码
也可以自己创建一个配置文件,比如 app.config.js
:
const path = require('path')module.exports = { // 一份配置文件可以同时管理多个 node.js 应用 // apps 是一个数组,每一项都是一个应用的配置 apps: [{ // 应用名称 name: "express-demo", // 应用入口文件 script: "./server.js", // 启动应用的模式, 有两种:cluster和fork,默认是fork exec_mode: 'cluster', // 创建应用实例的数量 instances: 'max', // 开启监听,当文件变化后自动重启应用 watch: true, // 忽略掉一些目录文件的变化。 // 由于把日志目录放到了项目路径下,一定要将其忽略,否则应用启动产生日志,pm2 监听到变化就会重启,重启又产生日志,就会进入死循环 ignore_watch: [ "node_modules", "logs" ], // 错误日志存放路径 err_file: path.resolve(__dirname, 'logs/error.log'), // 打印日志存放路径 out_file: path.resolve(__dirname, 'logs/out.log'), // 设置日志文件中每条日志前面的日期格式 log_date_format: "YYYY-MM-DD HH:mm:ss", }] }复制代码
让 pm2 使用配置文件来管理 node 应用:
pm2 start app.config.js复制代码
现在 pm2 管理的应用,会将日志放到项目目录下(默认是放到 pm2 的安装目录下),并且能监听文件的变化,自动重启服务。
更多有用的配置可以参考 PM2 官方文档,点此访问。
上面我们直接将 nodejs 项目的3000端口暴露了出去。一般我们都会使用 nginx 做一个代理转发,只对外暴露 80 端口。
首先服务器中需要安装 nginx ,有三种方式:
我这里的系统是 CentOS 8,已经更换了可用的 yum 源,可以直接安装 nginx。如果你的操作系统为 CentOS 7 或者其他发行版,可以搜索适合的安装方法。
使用 yum 安装:
# yum install -y nginx复制代码
然后启动 nginx:
# systemctl start nginx复制代码
打开浏览器访问服务器地址,可以看到 nginx 默认的主页:
为项目新建一个配置文件:
# vim /etc/nginx/conf.d/express.conf复制代码
监听80端口,将所有请求转发给服务器本地的3000端口的程序处理:
server { listen 80; server_name ironfan.site; location / { proxy_pass http://localhost:3000; } }复制代码
conf
目录下的配置文件,会被主配置文件 /etc/nginx/nginx.conf
加载:
修改完配置文件,一定要重启服务:
# systemctl restart nginx复制代码
然后本地打开浏览器,去掉原来的3000端口号,直接访问完整的 url:
到这里,就完成了接口转发的配置。从用户的角度出发,这个也叫反向代理。
首先我们比较系统的讲解了为何需要在 node.js 项目中开启多进程,以及两种实现方式:
child_process
模块的 fork
方法cluster 模块
的 fork
方法之后,又讲解了如何在 Linux 服务器中安装 node 环境,以及部署一个 node.js 项目的大致流程,并着重介绍了 pm2 的使用:
最后,讲解了使用 nginx 实现接口的代理转发,将用户请求转发到本地的3000端口的服务。
至此,我们完成了本文的目标,将一个 express 项目部署到服务器,并能稳定可靠的运行。
下篇文章,我们会使用 Github Actions 实现 CI/CD,让项目的部署更加便捷高效。
本文演示代码,已上传至 Github,点击访问。
更多node相关知识,请访问:nodejs 教程!
Atas ialah kandungan terperinci Artikel untuk membincangkan model berbilang proses Node dan penggunaan projek. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!