Node.js 的 dgram
模块提供了 UDP(用户数据报协议)实现,允许我们通过 Datagram Sockets 进行无连接通信。UDP 是一种无连接、面向数据报的协议,与 TCP 不同,它不提供传输保障,适合对速度有要求的应用场景,如流媒体、在线游戏等。
在 dgram
模块中,Datagram Sockets 主要用于发送和接收数据报。以下是 dgram
模块的相关属性、方法以及详细使用方式,涵盖各个方面。
dgram
模块首先,需要将 dgram
模块引入到你的项目中:
const dgram = require('dgram');
使用 dgram.createSocket()
来创建一个 UDP socket,它将返回一个 dgram.Socket
对象,用于接收或发送数据。
const socket = dgram.createSocket(type, [callback]);
type
:指定 socket 使用的协议类型。可以是 'udp4'
(IPv4)或 'udp6'
(IPv6)。callback
:可选的回调函数,当 socket 接收到消息时调用。const socket = dgram.createSocket('udp4');
在这个例子中,我们创建了一个基于 IPv4 的 UDP socket。
dgram.Socket
类继承自 EventEmitter
,提供了一些重要的事件,用于处理 UDP 通信过程中的不同状态和情况。
'message'
:当 socket 接收到消息时触发。
msg
:接收到的消息(Buffer 对象)。rinfo
:包含远程信息的对象(address
、family
、port
、size
)。socket.on('message', (msg, rinfo) => {
console.log(`Received message: ${msg} from ${rinfo.address}:${rinfo.port}`);
});
'listening'
:当 socket 准备好接收数据时触发。
socket.on('listening', () => {
const address = socket.address();
console.log(`Socket listening on ${address.address}:${address.port}`);
});
'error'
:当发生错误时触发。
err
:错误对象。socket.on('error', (err) => {
console.error(`Socket error:\n${err.stack}`);
socket.close();
});
'close'
:当 socket 被关闭时触发。
socket.on('close', () => {
console.log('Socket closed');
});
dgram.Socket
提供了多种方法来发送数据、关闭 socket 和控制其行为。
socket.bind([port], [address], [callback])
将 socket 绑定到指定的 port
和 address
。如果没有指定 port
,系统将分配一个随机端口。如果没有指定 address
,默认绑定到 '0.0.0.0'
或 '::'
(取决于使用的 IP 版本)。
port
:要绑定的端口号。address
:要绑定的 IP 地址(如 'localhost'
、'127.0.0.1'
、'0.0.0.0'
或 '::1'
)。callback
:绑定成功时调用的回调函数。socket.bind(12345, 'localhost', () => {
console.log('Socket bound to port 12345');
});
socket.send(msg, [offset], [length], port, address, [callback])
发送数据报到指定的 port
和 address
。
msg
:要发送的消息,可以是 Buffer
、字符串或数组。offset
:可选,指定从 msg
中读取数据的开始位置。length
:可选,要发送的字节长度。port
:目标主机的端口号。address
:目标主机的 IP 地址或域名。callback
:可选,数据发送后调用的回调函数。const message = Buffer.from('Hello UDP');
socket.send(message, 0, message.length, 12345, 'localhost', (err) => {
if (err) console.error('Send error:', err);
console.log('Message sent');
});
socket.close([callback])
关闭 socket。当 socket 被关闭时,'close'
事件将会触发。可选的 callback
将在关闭后调用。
socket.close(() => {
console.log('Socket closed');
});
socket.setBroadcast(flag)
启用或禁用广播消息。如果 flag
为 true
,socket 可以发送 UDP 广播消息。
socket.setBroadcast(true);
socket.setTTL(ttl)
设置 IP 数据报的生存时间(TTL,Time to Live),默认值为 64。ttl
是一个 1 到 255 的数字,表示该数据报在网络中的最大跳数。
socket.setTTL(128);
socket.setMulticastTTL(ttl)
设置组播数据报的 TTL(Time to Live)。ttl
是一个 1 到 255 的数字。
socket.setMulticastTTL(64);
socket.setMulticastLoopback(flag)
启用或禁用组播环回。如果 flag
为 true
,本地回送(loopback)将会启用,这意味着组播数据报也会发送到发送端的本地接口。
socket.setMulticastLoopback(true);
socket.addMembership(multicastAddress, [multicastInterface])
加入一个指定的组播地址。可选的 multicastInterface
是要使用的网络接口的 IP 地址。
socket.addMembership('239.255.255.250'); // 加入某个组播组
socket.dropMembership(multicastAddress, [multicastInterface])
离开一个组播地址。
socket.dropMembership('239.255.255.250'); // 离开某个组播组
socket.ref()
保持 Node.js 事件循环,阻止其退出。如果 socket 是唯一活跃的事件源,调用此方法将允许事件循环保持运行。
socket.ref();
socket.unref()
允许 Node.js 事件循环退出,即使此 socket 仍然打开。如果 socket 是唯一活跃的事件源,调用此方法将允许事件循环退出。
socket.unref();
UDP 广播允许将数据报发送到一个网络中的所有设备。
要发送广播消息,需要启用广播:
socket.setBroadcast(true);
然后发送到广播地址(如 255.255.255.255
或某个子网内的地址):
const message = Buffer.from('Broadcast message');
socket.send(message, 0, message.length, 41234, '255.255.255.255', (err) => {
if (err) console.error('Broadcast error:', err);
console.log('Broadcast message sent');
});
UDP 组播用于将消息发送给多个订阅特定组播地址的设备。组播地址的范围通常在 224.0.0.0
到 239.255.255.255
之间。
socket.addMembership('239.255.255.250');
socket.setMulticastTTL(128);
socket.setMulticastLoopback(true);
const dgram = require('dgram');
const server = dgram.createSocket('udp4');
server.on('message', (msg, rinfo) => {
console.log(`Server got: ${msg} from ${rinfo.address}:${rinfo.port}`);
});
server.on('listening', () => {
const address = server.address();
console.log(`Server listening on ${address.address}:${address.port}`);
});
server.bind(41234); // 服务器绑定端口
const dgram = require('dgram');
const client = dgram.createSocket('
udp4');
const message = Buffer.from('Hello, UDP server!');
client.send(message, 0, message.length, 41234, 'localhost', (err) => {
if (err) console.error('Send error:', err);
client.close();
});
UDP 的优势在于速度快、开销低,但它没有像 TCP 那样的可靠性保障,因此适合不需要可靠传输或有自己纠错机制的应用场景。
总结来说,dgram
模块提供了强大的工具来构建基于 UDP 的通信应用。