问答题783/1593两个 Node.js 进程如何通信?

难度:
2022-03-09 创建

参考答案:

两个 Node.js 进程之间如何进行通信呢?这里要分两种场景:

  1. 不同电脑上的两个 Node.js 进程间通信
  2. 同一台电脑上两个 Node.js 进程间通信

对于第一种场景,通常使用 TCP 或 HTTP 进行通信,而对于第二种场景,又分为两种子场景:

  1. Node.js 进程和自己创建的 Node.js 子进程通信
  2. Node.js 进程和另外不相关的 Node.js 进程通信

前者可以使用内置的 IPC 通信通道,后者可以使用自定义管道,接下来进行详细介绍:

不同电脑上的两个 Node.js 进程间通信

要想进行通信,首先得搞清楚如何标识网络中的进程?网络层的 ip 地址可以唯一标识网络中的主机,而传输层的协议和端口可以唯一标识主机中的应用程序(进程),这样利用三元组(ip 地址,协议,端口)就可以标识网络的进程了。

使用 TCP 套接字

TCP 套接字(socket)是一种基于 TCP/IP 协议的通信方式,可以让通过网络连接的计算机上的进程进行通信。一个作为 server 另一个作为 client,server.js 代码如下:

1const net = require('net') 2const server = net.createServer(socket => { 3 console.log('socket connected') 4 socket.on('close', () => console.log('socket disconnected')) 5 socket.on('error', err => console.error(err.message)) 6 socket.on('data', data => { 7 console.log(`receive: ${data}`) 8 socket.write(data) 9 console.log(`send: ${data}`) 10 }) 11}) 12server.listen(8888) 13

client.js 代码:

1const net = require('net') 2const client = net.connect(8888, '192.168.10.105') 3 4client.on('connect', () => console.log('connected.')) 5client.on('data', data => console.log(`receive: ${data}`)) 6client.on('end', () => console.log('disconnected.')) 7client.on('error', err => console.error(err.message)) 8 9setInterval(() => { 10 const msg = 'hello' 11 console.log(`send: ${msg}`) 12 client.write(msg) 13}, 3000) 14

运行效果:

1$ node server.js 2client connected 3receive: hello 4send: hello 5 6$ node client.js 7connect to server 8send: hello 9receive: hello 10

使用 HTTP 协议

因为 HTTP 协议也是基于 TCP 的,所以从通信角度看,这种方式本质上并无区别,只是封装了上层协议。server.js 代码为:

1const http = require('http') 2http.createServer((req, res) => res.end(req.url)).listen(8888) 3

client.js 代码:

1const http = require('http') 2const options = { 3 hostname: '192.168.10.105', 4 port: 8888, 5 path: '/hello', 6 method: 'GET', 7} 8const req = http.request(options, res => { 9 console.log(`statusCode: ${res.statusCode}`) 10 res.on('data', d => process.stdout.write(d)) 11}) 12req.on('error', error => console.error(error)) 13req.end() 14

运行效果:

1$ node server.js 2url /hello 3 4$ node client.js 5statusCode: 200 6hello 7

同一台电脑上两个 Node.js 进程间通信

虽然网络 socket 也可用于同一台主机的进程间通讯(通过 loopback 地址 127.0.0.1),但是这种方式需要经过网络协议栈、需要打包拆包、计算校验和、维护序号和应答等,就是为网络通讯设计的,而同一台电脑上的两个进程可以有更高效的通信方式,即 IPC(Inter-Process Communication),在 unix 上具体的实现方式为 unix domain socket,这是服务器端和客户端之间通过本地打开的套接字文件进行通信的一种方法,与 TCP 通信不同,通信时指定本地文件,因此不进行域解析和外部通信,所以比 TCP 快,在同一台主机的传输速度是 TCP 的两倍。

使用内置 IPC 通道

如果是跟自己创建的子进程通信,是非常方便的,child_process模块中的 fork 方法自带通信机制,无需关注底层细节,例如父进程 parent.js 代码:

1const fork = require("child_process").fork 2const path = require("path") 3const child = fork(path.resolve("child.js"), [], { stdio: "inherit" }); 4child.on("message", (message) => { 5 console.log("message from child:", message) 6 child.send("hi") 7})

子进程 child.js 代码:

1process.on("message", (message) => { 2 console.log("message from parent:", message); 3}) 4 5if (process.send) { 6 setInterval(() => process.send("hello"), 3000) 7} 8

运行效果如下:

1$ node parent.js 2message from child: hello 3message from parent: hi 4message from child: hello 5message from parent: hi 6

使用自定义管道

如果是两个独立的 Node.js 进程,如何建立通信通道呢?在 Windows 上可以使用命名管道(Named PIPE),在 unix 上可以使用 unix domain socket,也是一个作为 server,另外一个作为 client,其中 server.js 代码如下:

1const net = require('net') 2const fs = require('fs') 3 4const pipeFile = process.platform === 'win32' ? '\\\\.\\pipe\\mypip' : '/tmp/unix.sock' 5 6const server = net.createServer(connection => { 7 console.log('socket connected.') 8 connection.on('close', () => console.log('disconnected.')) 9 connection.on('data', data => { 10 console.log(`receive: ${data}`) 11 connection.write(data) 12 console.log(`send: ${data}`) 13 }) 14 connection.on('error', err => console.error(err.message)) 15}) 16 17try { 18 fs.unlinkSync(pipeFile) 19} catch (error) {} 20 21server.listen(pipeFile)

client.js 代码如下:

1const net = require('net') 2 3const pipeFile = process.platform === 'win32' ? '\\\\.\\pipe\\mypip' : '/tmp/unix.sock' 4 5const client = net.connect(pipeFile) 6client.on('connect', () => console.log('connected.')) 7client.on('data', data => console.log(`receive: ${data}`)) 8client.on('end', () => console.log('disconnected.')) 9client.on('error', err => console.error(err.message)) 10 11setInterval(() => { 12 const msg = 'hello' 13 console.log(`send: ${msg}`) 14 client.write(msg) 15}, 3000) 16 17

运行效果:

1$ node server.js 2socket connected. 3receive: hello 4send: hello 5 6$ node client.js 7connected. 8send: hello 9receive: hello 10

最近更新时间:2024-08-10

赞赏支持

预览

题库维护不易,您的支持就是我们最大的动力!